/* * Copyright (C) 2008 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.settings.accounts; import static android.app.admin.DevicePolicyResources.Strings.Settings.ACCESSIBILITY_PERSONAL_ACCOUNT_TITLE; import static android.app.admin.DevicePolicyResources.Strings.Settings.ACCESSIBILITY_WORK_ACCOUNT_TITLE; import android.accounts.Account; import android.accounts.AccountManager; import android.app.Activity; import android.app.Dialog; import android.app.admin.DevicePolicyManager; import android.app.settings.SettingsEnums; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.IntentSender; import android.content.SyncAdapterType; import android.content.SyncInfo; import android.content.SyncStatusInfo; import android.content.pm.PackageManager; import android.content.pm.ProviderInfo; import android.content.pm.UserInfo; import android.os.Binder; import android.os.Bundle; import android.os.UserHandle; import android.os.UserManager; import android.text.TextUtils; import android.text.format.DateUtils; import android.util.Log; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import androidx.annotation.VisibleForTesting; import androidx.appcompat.app.AlertDialog; import androidx.preference.Preference; import com.android.settings.R; import com.android.settings.Utils; import com.android.settings.widget.EntityHeaderController; import com.android.settingslib.widget.FooterPreference; import com.google.android.collect.Lists; import java.util.ArrayList; import java.util.Date; import java.util.HashMap; import java.util.List; public class AccountSyncSettings extends AccountPreferenceBase { public static final String ACCOUNT_KEY = "account"; private static final int MENU_SYNC_NOW_ID = Menu.FIRST; private static final int MENU_SYNC_CANCEL_ID = Menu.FIRST + 1; private static final int CANT_DO_ONETIME_SYNC_DIALOG = 102; private static final String UID_REQUEST_KEY = "uid_request_code"; private Account mAccount; private ArrayList mInvisibleAdapters = Lists.newArrayList(); private HashMap mUidRequestCodeMap = new HashMap<>(); @Override public Dialog onCreateDialog(final int id) { Dialog dialog = null; if (id == CANT_DO_ONETIME_SYNC_DIALOG) { dialog = new AlertDialog.Builder(getActivity()) .setTitle(R.string.cant_sync_dialog_title) .setMessage(R.string.cant_sync_dialog_message) .setPositiveButton(android.R.string.ok, null) .create(); } return dialog; } @Override public int getMetricsCategory() { return SettingsEnums.ACCOUNTS_ACCOUNT_SYNC; } @Override public int getDialogMetricsCategory(int dialogId) { switch (dialogId) { case CANT_DO_ONETIME_SYNC_DIALOG: return SettingsEnums.DIALOG_ACCOUNT_SYNC_CANNOT_ONETIME_SYNC; default: return 0; } } @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); addPreferencesFromResource(R.xml.account_sync_settings); getPreferenceScreen().setOrderingAsAdded(false); setAccessibilityTitle(); } @Override public void onSaveInstanceState(Bundle outState) { super.onSaveInstanceState(outState); if (!mUidRequestCodeMap.isEmpty()) { outState.putSerializable(UID_REQUEST_KEY, mUidRequestCodeMap); } } @Override public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); Bundle arguments = getArguments(); if (arguments == null) { Log.e(TAG, "No arguments provided when starting intent. ACCOUNT_KEY needed."); finish(); return; } mAccount = arguments.getParcelable(ACCOUNT_KEY); if (!accountExists(mAccount)) { Log.e(TAG, "Account provided does not exist: " + mAccount); finish(); return; } if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, "Got account: " + mAccount); } final Activity activity = getActivity(); final Preference pref = EntityHeaderController .newInstance(activity, this, null /* header */) .setIcon(getDrawableForType(mAccount.type)) .setLabel(mAccount.name) .setSummary(getLabelForType(mAccount.type)) .done(getPrefContext()); pref.setOrder(0); getPreferenceScreen().addPreference(pref); if (savedInstanceState != null && savedInstanceState.containsKey(UID_REQUEST_KEY)) { mUidRequestCodeMap = (HashMap) savedInstanceState.getSerializable( UID_REQUEST_KEY); } } private void setAccessibilityTitle() { final UserManager um = (UserManager) getSystemService(Context.USER_SERVICE); UserInfo user = um.getUserInfo(mUserHandle.getIdentifier()); boolean isWorkProfile = user != null ? user.isManagedProfile() : false; CharSequence currentTitle = getActivity().getTitle(); DevicePolicyManager devicePolicyManager = getSystemService(DevicePolicyManager.class); String accessibilityTitle = isWorkProfile ? devicePolicyManager.getResources().getString( ACCESSIBILITY_WORK_ACCOUNT_TITLE, () -> getString(R.string.accessibility_work_account_title, currentTitle), currentTitle) : devicePolicyManager.getResources().getString( ACCESSIBILITY_PERSONAL_ACCOUNT_TITLE, () -> getString( R.string.accessibility_personal_account_title, currentTitle), currentTitle); getActivity().setTitle(Utils.createAccessibleSequence(currentTitle, accessibilityTitle)); } @Override public void onResume() { mAuthenticatorHelper.listenToAccountUpdates(); updateAuthDescriptions(); onAccountsUpdate(Binder.getCallingUserHandle()); super.onResume(); } @Override public void onPause() { super.onPause(); mAuthenticatorHelper.stopListeningToAccountUpdates(); } private void addSyncStateSwitch(Account account, String authority, String packageName, int uid) { SyncStateSwitchPreference item = (SyncStateSwitchPreference) getCachedPreference(authority); if (item == null) { item = new SyncStateSwitchPreference(getPrefContext(), account, authority, packageName, uid); getPreferenceScreen().addPreference(item); } else { item.setup(account, authority, packageName, uid); } final PackageManager packageManager = getPackageManager(); item.setPersistent(false); final ProviderInfo providerInfo = packageManager.resolveContentProviderAsUser( authority, 0, mUserHandle.getIdentifier()); if (providerInfo == null) { return; } final CharSequence providerLabel = providerInfo.loadLabel(packageManager); if (TextUtils.isEmpty(providerLabel)) { Log.e(TAG, "Provider needs a label for authority '" + authority + "'"); return; } item.setTitle(providerLabel); item.setKey(authority); } @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { MenuItem syncNow = menu.add(0, MENU_SYNC_NOW_ID, 0, getString(R.string.sync_menu_sync_now)); MenuItem syncCancel = menu.add(0, MENU_SYNC_CANCEL_ID, 0, getString(R.string.sync_menu_sync_cancel)); syncNow.setShowAsAction(MenuItem.SHOW_AS_ACTION_NEVER | MenuItem.SHOW_AS_ACTION_WITH_TEXT); syncCancel.setShowAsAction(MenuItem.SHOW_AS_ACTION_NEVER | MenuItem.SHOW_AS_ACTION_WITH_TEXT); super.onCreateOptionsMenu(menu, inflater); } @Override public void onPrepareOptionsMenu(Menu menu) { super.onPrepareOptionsMenu(menu); // Note that this also counts accounts that are not currently displayed boolean syncActive = !ContentResolver.getCurrentSyncsAsUser( mUserHandle.getIdentifier()).isEmpty(); menu.findItem(MENU_SYNC_NOW_ID).setVisible(!syncActive).setEnabled(enabledSyncNowMenu()); menu.findItem(MENU_SYNC_CANCEL_ID).setVisible(syncActive); } @Override public boolean onOptionsItemSelected(MenuItem item) { switch (item.getItemId()) { case MENU_SYNC_NOW_ID: startSyncForEnabledProviders(); return true; case MENU_SYNC_CANCEL_ID: cancelSyncForEnabledProviders(); return true; } return super.onOptionsItemSelected(item); } @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { if (resultCode == Activity.RESULT_OK) { final int count = getPreferenceScreen().getPreferenceCount(); for (int i = 0; i < count; i++) { Preference preference = getPreferenceScreen().getPreference(i); if (preference instanceof SyncStateSwitchPreference) { SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) preference; if (getRequestCodeByUid(syncPref.getUid()) == requestCode) { onPreferenceTreeClick(syncPref); return; } } } } } @Override public boolean onPreferenceTreeClick(Preference preference) { if (getActivity() == null) { return false; } if (preference instanceof SyncStateSwitchPreference) { SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) preference; final String authority = syncPref.getAuthority(); if (TextUtils.isEmpty(authority)) { return false; } final Account account = syncPref.getAccount(); final int userId = mUserHandle.getIdentifier(); final String packageName = syncPref.getPackageName(); boolean syncAutomatically = ContentResolver.getSyncAutomaticallyAsUser(account, authority, userId); if (syncPref.isOneTimeSyncMode()) { // If the sync adapter doesn't have access to the account we either // request access by starting an activity if possible or kick off the // sync which will end up posting an access request notification. if (requestAccountAccessIfNeeded(packageName)) { return true; } requestOrCancelSync(account, authority, true); } else { boolean syncOn = syncPref.isChecked(); boolean oldSyncState = syncAutomatically; if (syncOn != oldSyncState) { // Toggling this switch triggers sync but we may need a user approval. // If the sync adapter doesn't have access to the account we either // request access by starting an activity if possible or kick off the // sync which will end up posting an access request notification. if (syncOn && requestAccountAccessIfNeeded(packageName)) { return true; } // if we're enabling sync, this will request a sync as well ContentResolver.setSyncAutomaticallyAsUser(account, authority, syncOn, userId); // if the primary sync switch is off, the request above will // get dropped. when the user clicks on this toggle, // we want to force the sync, however. if (!ContentResolver.getMasterSyncAutomaticallyAsUser(userId) || !syncOn) { requestOrCancelSync(account, authority, syncOn); } } } return true; } else { return super.onPreferenceTreeClick(preference); } } private boolean requestAccountAccessIfNeeded(String packageName) { if (packageName == null) { return false; } final int uid; try { uid = getContext().getPackageManager().getPackageUidAsUser( packageName, mUserHandle.getIdentifier()); } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "Invalid sync ", e); return false; } AccountManager accountManager = getContext().getSystemService(AccountManager.class); if (!accountManager.hasAccountAccess(mAccount, packageName, mUserHandle)) { IntentSender intent = accountManager.createRequestAccountAccessIntentSenderAsUser( mAccount, packageName, mUserHandle); if (intent != null) { try { final int requestCode = addUidAndGenerateRequestCode(uid); startIntentSenderForResult(intent, requestCode, null /* fillInIntent */, 0, 0, 0, null /* options */); return true; } catch (IntentSender.SendIntentException e) { Log.e(TAG, "Error requesting account access", e); } } } return false; } private void startSyncForEnabledProviders() { requestOrCancelSyncForEnabledProviders(true /* start them */); final Activity activity = getActivity(); if (activity != null) { activity.invalidateOptionsMenu(); } } private void cancelSyncForEnabledProviders() { requestOrCancelSyncForEnabledProviders(false /* cancel them */); final Activity activity = getActivity(); if (activity != null) { activity.invalidateOptionsMenu(); } } private void requestOrCancelSyncForEnabledProviders(boolean startSync) { // sync everything that the user has enabled int count = getPreferenceScreen().getPreferenceCount(); for (int i = 0; i < count; i++) { Preference pref = getPreferenceScreen().getPreference(i); if (!(pref instanceof SyncStateSwitchPreference)) { continue; } SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) pref; if (!syncPref.isChecked()) { continue; } requestOrCancelSync(syncPref.getAccount(), syncPref.getAuthority(), startSync); } // plus whatever the system needs to sync, e.g., invisible sync adapters if (mAccount != null) { for (SyncAdapterType syncAdapter : mInvisibleAdapters) { requestOrCancelSync(mAccount, syncAdapter.authority, startSync); } } } private void requestOrCancelSync(Account account, String authority, boolean flag) { if (flag) { Bundle extras = new Bundle(); extras.putBoolean(ContentResolver.SYNC_EXTRAS_MANUAL, true); ContentResolver.requestSyncAsUser(account, authority, mUserHandle.getIdentifier(), extras); } else { ContentResolver.cancelSyncAsUser(account, authority, mUserHandle.getIdentifier()); } } private boolean isSyncing(List currentSyncs, Account account, String authority) { for (SyncInfo syncInfo : currentSyncs) { if (syncInfo.account.equals(account) && syncInfo.authority.equals(authority)) { return true; } } return false; } @Override protected void onSyncStateUpdated() { if (!isResumed()) return; setFeedsState(); final Activity activity = getActivity(); if (activity != null) { activity.invalidateOptionsMenu(); } } private void setFeedsState() { // iterate over all the preferences, setting the state properly for each Date date = new Date(); final int userId = mUserHandle.getIdentifier(); List currentSyncs = ContentResolver.getCurrentSyncsAsUser(userId); boolean syncIsFailing = false; // Refresh the sync status switches - some syncs may have become active. updateAccountSwitches(); for (int i = 0, count = getPreferenceScreen().getPreferenceCount(); i < count; i++) { Preference pref = getPreferenceScreen().getPreference(i); if (!(pref instanceof SyncStateSwitchPreference)) { continue; } SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) pref; String authority = syncPref.getAuthority(); Account account = syncPref.getAccount(); SyncStatusInfo status = ContentResolver.getSyncStatusAsUser(account, authority, userId); boolean syncEnabled = ContentResolver.getSyncAutomaticallyAsUser(account, authority, userId); boolean authorityIsPending = status == null ? false : status.pending; boolean initialSync = status == null ? false : status.initialize; boolean activelySyncing = isSyncing(currentSyncs, account, authority); boolean lastSyncFailed = status != null && status.lastFailureTime != 0 && status.getLastFailureMesgAsInt(0) != ContentResolver.SYNC_ERROR_SYNC_ALREADY_IN_PROGRESS; if (!syncEnabled) lastSyncFailed = false; if (lastSyncFailed && !activelySyncing && !authorityIsPending) { syncIsFailing = true; } if (Log.isLoggable(TAG, Log.DEBUG)) { Log.d(TAG, "Update sync status: " + account + " " + authority + " active = " + activelySyncing + " pend =" + authorityIsPending); } final long successEndTime = (status == null) ? 0 : status.lastSuccessTime; if (!syncEnabled) { syncPref.setSummary(R.string.sync_disabled); } else if (activelySyncing) { syncPref.setSummary(R.string.sync_in_progress); } else if (successEndTime != 0) { date.setTime(successEndTime); final String timeString = formatSyncDate(getContext(), date); syncPref.setSummary(getResources().getString(R.string.last_synced, timeString)); } else { syncPref.setSummary(""); } int syncState = ContentResolver.getIsSyncableAsUser(account, authority, userId); syncPref.setActive(activelySyncing && (syncState >= 0) && !initialSync); syncPref.setPending(authorityIsPending && (syncState >= 0) && !initialSync); syncPref.setFailed(lastSyncFailed); final boolean oneTimeSyncMode = !ContentResolver.getMasterSyncAutomaticallyAsUser( userId); syncPref.setOneTimeSyncMode(oneTimeSyncMode); syncPref.setChecked(oneTimeSyncMode || syncEnabled); } if (syncIsFailing) { getPreferenceScreen().addPreference(new FooterPreference.Builder( getActivity()).setTitle(R.string.sync_is_failing).build()); } } @Override public void onAccountsUpdate(final UserHandle userHandle) { super.onAccountsUpdate(userHandle); if (!accountExists(mAccount)) { // The account was deleted finish(); return; } updateAccountSwitches(); onSyncStateUpdated(); } private boolean accountExists(Account account) { if (account == null) return false; Account[] accounts = AccountManager.get(getActivity()).getAccountsByTypeAsUser( account.type, mUserHandle); final int count = accounts.length; for (int i = 0; i < count; i++) { if (accounts[i].equals(account)) { return true; } } return false; } private void updateAccountSwitches() { mInvisibleAdapters.clear(); SyncAdapterType[] syncAdapters = ContentResolver.getSyncAdapterTypesAsUser( mUserHandle.getIdentifier()); ArrayList authorities = new ArrayList<>(); for (int i = 0, n = syncAdapters.length; i < n; i++) { final SyncAdapterType sa = syncAdapters[i]; // Only keep track of sync adapters for this account if (!sa.accountType.equals(mAccount.type)) continue; if (sa.isUserVisible()) { if (Log.isLoggable(TAG, Log.DEBUG)) { Log.d(TAG, "updateAccountSwitches: added authority " + sa.authority + " to accountType " + sa.accountType); } authorities.add(sa); } else { // keep track of invisible sync adapters, so sync now forces // them to sync as well. mInvisibleAdapters.add(sa); } } if (Log.isLoggable(TAG, Log.DEBUG)) { Log.d(TAG, "looking for sync adapters that match account " + mAccount); } cacheRemoveAllPrefs(getPreferenceScreen()); getCachedPreference(EntityHeaderController.PREF_KEY_APP_HEADER); for (int j = 0, m = authorities.size(); j < m; j++) { final SyncAdapterType syncAdapter = authorities.get(j); // We could check services here.... int syncState = ContentResolver.getIsSyncableAsUser(mAccount, syncAdapter.authority, mUserHandle.getIdentifier()); if (Log.isLoggable(TAG, Log.DEBUG)) { Log.d(TAG, " found authority " + syncAdapter.authority + " " + syncState); } if (syncState > 0) { final int uid; try { uid = getContext().getPackageManager().getPackageUidAsUser( syncAdapter.getPackageName(), mUserHandle.getIdentifier()); addSyncStateSwitch(mAccount, syncAdapter.authority, syncAdapter.getPackageName(), uid); } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "No uid for package" + syncAdapter.getPackageName(), e); } } } removeCachedPrefs(getPreferenceScreen()); } @Override public int getHelpResource() { return R.string.help_url_accounts; } @VisibleForTesting boolean enabledSyncNowMenu() { boolean enabled = false; for (int i = 0, count = getPreferenceScreen().getPreferenceCount(); i < count; i++) { final Preference pref = getPreferenceScreen().getPreference(i); if (!(pref instanceof SyncStateSwitchPreference)) { continue; } final SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) pref; if (syncPref.isChecked()) { enabled = true; break; } } return enabled; } private static String formatSyncDate(Context context, Date date) { return DateUtils.formatDateTime(context, date.getTime(), DateUtils.FORMAT_SHOW_DATE | DateUtils.FORMAT_SHOW_YEAR | DateUtils.FORMAT_SHOW_TIME); } private int addUidAndGenerateRequestCode(int uid) { if (mUidRequestCodeMap.containsKey(uid)) { return mUidRequestCodeMap.get(uid); } final int requestCode = mUidRequestCodeMap.size() + 1; mUidRequestCodeMap.put(uid, requestCode); return requestCode; } private int getRequestCodeByUid(int uid) { if (!mUidRequestCodeMap.containsKey(uid)) { return -1; } return mUidRequestCodeMap.get(uid); } }