blob: 67c931f9e4768be04148111e1970b6468b50c70d [file] [log] [blame]
/*
* Copyright (C) 2021 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.server.locales;
import static android.os.UserHandle.USER_NULL;
import static com.android.server.locales.LocaleManagerService.DEBUG;
import android.annotation.NonNull;
import android.annotation.UserIdInt;
import android.app.backup.BackupManager;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.HandlerThread;
import android.os.LocaleList;
import android.os.RemoteException;
import android.os.UserHandle;
import android.text.TextUtils;
import android.util.Slog;
import android.util.SparseArray;
import android.util.Xml;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.util.XmlUtils;
import com.android.modules.utils.TypedXmlPullParser;
import com.android.modules.utils.TypedXmlSerializer;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.nio.charset.StandardCharsets;
import java.time.Clock;
import java.time.Duration;
import java.util.HashMap;
/**
* Helper class for managing backup and restore of app-specific locales.
*/
class LocaleManagerBackupHelper {
private static final String TAG = "LocaleManagerBkpHelper"; // must be < 23 chars
// Tags and attributes for xml.
private static final String LOCALES_XML_TAG = "locales";
private static final String PACKAGE_XML_TAG = "package";
private static final String ATTR_PACKAGE_NAME = "name";
private static final String ATTR_LOCALES = "locales";
private static final String ATTR_CREATION_TIME_MILLIS = "creationTimeMillis";
private static final String SYSTEM_BACKUP_PACKAGE_KEY = "android";
// Stage data would be deleted on reboot since it's stored in memory. So it's retained until
// retention period OR next reboot, whichever happens earlier.
private static final Duration STAGE_DATA_RETENTION_PERIOD = Duration.ofDays(3);
private final LocaleManagerService mLocaleManagerService;
private final PackageManager mPackageManager;
private final Clock mClock;
private final Context mContext;
private final Object mStagedDataLock = new Object();
// Staged data map keyed by user-id to handle multi-user scenario / work profiles. We are using
// SparseArray because it is more memory-efficient than a HashMap.
private final SparseArray<StagedData> mStagedData;
private final BroadcastReceiver mUserMonitor;
LocaleManagerBackupHelper(LocaleManagerService localeManagerService,
PackageManager packageManager, HandlerThread broadcastHandlerThread) {
this(localeManagerService.mContext, localeManagerService, packageManager, Clock.systemUTC(),
new SparseArray<>(), broadcastHandlerThread);
}
@VisibleForTesting LocaleManagerBackupHelper(Context context,
LocaleManagerService localeManagerService,
PackageManager packageManager, Clock clock, SparseArray<StagedData> stagedData,
HandlerThread broadcastHandlerThread) {
mContext = context;
mLocaleManagerService = localeManagerService;
mPackageManager = packageManager;
mClock = clock;
mStagedData = stagedData;
mUserMonitor = new UserMonitor();
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_USER_REMOVED);
context.registerReceiverAsUser(mUserMonitor, UserHandle.ALL, filter,
null, broadcastHandlerThread.getThreadHandler());
}
@VisibleForTesting
BroadcastReceiver getUserMonitor() {
return mUserMonitor;
}
/**
* @see LocaleManagerInternal#getBackupPayload(int userId)
*/
public byte[] getBackupPayload(int userId) {
if (DEBUG) {
Slog.d(TAG, "getBackupPayload invoked for user id " + userId);
}
synchronized (mStagedDataLock) {
cleanStagedDataForOldEntriesLocked();
}
HashMap<String, String> pkgStates = new HashMap<>();
for (ApplicationInfo appInfo : mPackageManager.getInstalledApplicationsAsUser(
PackageManager.ApplicationInfoFlags.of(0), userId)) {
try {
LocaleList appLocales = mLocaleManagerService.getApplicationLocales(
appInfo.packageName,
userId);
// Backup locales only for apps which do have app-specific overrides.
if (!appLocales.isEmpty()) {
if (DEBUG) {
Slog.d(TAG, "Add package=" + appInfo.packageName + " locales="
+ appLocales.toLanguageTags() + " to backup payload");
}
pkgStates.put(appInfo.packageName, appLocales.toLanguageTags());
}
} catch (RemoteException | IllegalArgumentException e) {
Slog.e(TAG, "Exception when getting locales for package: " + appInfo.packageName,
e);
}
}
if (pkgStates.isEmpty()) {
if (DEBUG) {
Slog.d(TAG, "Final payload=null");
}
// Returning null here will ensure deletion of the entry for LMS from the backup data.
return null;
}
final ByteArrayOutputStream out = new ByteArrayOutputStream();
try {
writeToXml(out, pkgStates);
} catch (IOException e) {
Slog.e(TAG, "Could not write to xml for backup ", e);
return null;
}
if (DEBUG) {
try {
Slog.d(TAG, "Final payload=" + out.toString("UTF-8"));
} catch (UnsupportedEncodingException e) {
Slog.w(TAG, "Could not encode payload to UTF-8", e);
}
}
return out.toByteArray();
}
private void cleanStagedDataForOldEntriesLocked() {
for (int i = 0; i < mStagedData.size(); i++) {
int userId = mStagedData.keyAt(i);
StagedData stagedData = mStagedData.get(userId);
if (stagedData.mCreationTimeMillis
< mClock.millis() - STAGE_DATA_RETENTION_PERIOD.toMillis()) {
deleteStagedDataLocked(userId);
}
}
}
/**
* @see LocaleManagerInternal#stageAndApplyRestoredPayload(byte[] payload, int userId)
*/
public void stageAndApplyRestoredPayload(byte[] payload, int userId) {
if (DEBUG) {
Slog.d(TAG, "stageAndApplyRestoredPayload user=" + userId + " payload="
+ (payload != null ? new String(payload, StandardCharsets.UTF_8) : null));
}
if (payload == null) {
Slog.e(TAG, "stageAndApplyRestoredPayload: no payload to restore for user " + userId);
return;
}
final ByteArrayInputStream inputStream = new ByteArrayInputStream(payload);
HashMap<String, String> pkgStates;
try {
// Parse the input blob into a list of BackupPackageState.
final TypedXmlPullParser parser = Xml.newFastPullParser();
parser.setInput(inputStream, StandardCharsets.UTF_8.name());
XmlUtils.beginDocument(parser, LOCALES_XML_TAG);
pkgStates = readFromXml(parser);
} catch (IOException | XmlPullParserException e) {
Slog.e(TAG, "Could not parse payload ", e);
return;
}
// We need a lock here to prevent race conditions when accessing the stage file.
// It might happen that a restore was triggered (manually using bmgr cmd) and at the same
// time a new package is added. We want to ensure that both these operations aren't
// performed simultaneously.
synchronized (mStagedDataLock) {
// Backups for apps which are yet to be installed.
StagedData stagedData = new StagedData(mClock.millis(), new HashMap<>());
for (String pkgName : pkgStates.keySet()) {
String languageTags = pkgStates.get(pkgName);
// Check if the application is already installed for the concerned user.
if (isPackageInstalledForUser(pkgName, userId)) {
// Don't apply the restore if the locales have already been set for the app.
checkExistingLocalesAndApplyRestore(pkgName, languageTags, userId);
} else {
// Stage the data if the app isn't installed.
stagedData.mPackageStates.put(pkgName, languageTags);
if (DEBUG) {
Slog.d(TAG, "Add locales=" + languageTags
+ " package=" + pkgName + " for lazy restore.");
}
}
}
if (!stagedData.mPackageStates.isEmpty()) {
mStagedData.put(userId, stagedData);
}
}
}
/**
* Notifies the backup manager to include the "android" package in the next backup pass.
*/
public void notifyBackupManager() {
BackupManager.dataChanged(SYSTEM_BACKUP_PACKAGE_KEY);
}
/**
* <p><b>Note:</b> This is invoked by service's common monitor
* {@link LocaleManagerServicePackageMonitor#onPackageAdded} when a new package is
* added on device.
*/
void onPackageAdded(String packageName, int uid) {
try {
synchronized (mStagedDataLock) {
cleanStagedDataForOldEntriesLocked();
int userId = UserHandle.getUserId(uid);
if (mStagedData.contains(userId)) {
// Perform lazy restore only if the staged data exists.
doLazyRestoreLocked(packageName, userId);
}
}
} catch (Exception e) {
Slog.e(TAG, "Exception in onPackageAdded.", e);
}
}
/**
* <p><b>Note:</b> This is invoked by service's common monitor
* {@link LocaleManagerServicePackageMonitor#onPackageDataCleared} when a package's data
* is cleared.
*/
void onPackageDataCleared() {
try {
notifyBackupManager();
} catch (Exception e) {
Slog.e(TAG, "Exception in onPackageDataCleared.", e);
}
}
/**
* <p><b>Note:</b> This is invoked by service's common monitor
* {@link LocaleManagerServicePackageMonitor#onPackageRemoved} when a package is removed
* from device.
*/
void onPackageRemoved() {
try {
notifyBackupManager();
} catch (Exception e) {
Slog.e(TAG, "Exception in onPackageRemoved.", e);
}
}
private boolean isPackageInstalledForUser(String packageName, int userId) {
PackageInfo pkgInfo = null;
try {
pkgInfo = mContext.getPackageManager().getPackageInfoAsUser(
packageName, /* flags= */ 0, userId);
} catch (PackageManager.NameNotFoundException e) {
if (DEBUG) {
Slog.d(TAG, "Could not get package info for " + packageName, e);
}
}
return pkgInfo != null;
}
/**
* Checks if locales already exist for the application and applies the restore accordingly.
* <p>
* The user might change the locales for an application before the restore is applied. In this
* case, we want to keep the user settings and discard the restore.
*/
private void checkExistingLocalesAndApplyRestore(@NonNull String pkgName,
@NonNull String languageTags, int userId) {
try {
LocaleList currLocales = mLocaleManagerService.getApplicationLocales(
pkgName,
userId);
if (!currLocales.isEmpty()) {
return;
}
} catch (RemoteException e) {
Slog.e(TAG, "Could not check for current locales before restoring", e);
}
// Restore the locale immediately
try {
mLocaleManagerService.setApplicationLocales(pkgName, userId,
LocaleList.forLanguageTags(languageTags));
if (DEBUG) {
Slog.d(TAG, "Restored locales=" + languageTags + " for package=" + pkgName);
}
} catch (RemoteException | IllegalArgumentException e) {
Slog.e(TAG, "Could not restore locales for " + pkgName, e);
}
}
private void deleteStagedDataLocked(@UserIdInt int userId) {
mStagedData.remove(userId);
}
/**
* Parses the backup data from the serialized xml input stream.
*/
private @NonNull HashMap<String, String> readFromXml(XmlPullParser parser)
throws IOException, XmlPullParserException {
HashMap<String, String> packageStates = new HashMap<>();
int depth = parser.getDepth();
while (XmlUtils.nextElementWithin(parser, depth)) {
if (parser.getName().equals(PACKAGE_XML_TAG)) {
String packageName = parser.getAttributeValue(/* namespace= */ null,
ATTR_PACKAGE_NAME);
String languageTags = parser.getAttributeValue(/* namespace= */ null, ATTR_LOCALES);
if (!TextUtils.isEmpty(packageName) && !TextUtils.isEmpty(languageTags)) {
packageStates.put(packageName, languageTags);
}
}
}
return packageStates;
}
/**
* Converts the list of app backup data into a serialized xml stream.
*/
private static void writeToXml(OutputStream stream, @NonNull HashMap<String, String> pkgStates)
throws IOException {
if (pkgStates.isEmpty()) {
// No need to write anything at all if pkgStates is empty.
return;
}
TypedXmlSerializer out = Xml.newFastSerializer();
out.setOutput(stream, StandardCharsets.UTF_8.name());
out.startDocument(/* encoding= */ null, /* standalone= */ true);
out.startTag(/* namespace= */ null, LOCALES_XML_TAG);
for (String pkg : pkgStates.keySet()) {
out.startTag(/* namespace= */ null, PACKAGE_XML_TAG);
out.attribute(/* namespace= */ null, ATTR_PACKAGE_NAME, pkg);
out.attribute(/* namespace= */ null, ATTR_LOCALES, pkgStates.get(pkg));
out.endTag(/*namespace= */ null, PACKAGE_XML_TAG);
}
out.endTag(/* namespace= */ null, LOCALES_XML_TAG);
out.endDocument();
}
static class StagedData {
final long mCreationTimeMillis;
final HashMap<String, String> mPackageStates;
StagedData(long creationTimeMillis, HashMap<String, String> pkgStates) {
mCreationTimeMillis = creationTimeMillis;
mPackageStates = pkgStates;
}
}
/**
* Broadcast listener to capture user removed event.
*
* <p>The stage data is deleted when a user is removed.
*/
private final class UserMonitor extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
try {
String action = intent.getAction();
if (action.equals(Intent.ACTION_USER_REMOVED)) {
final int userId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, USER_NULL);
synchronized (mStagedDataLock) {
deleteStagedDataLocked(userId);
}
}
} catch (Exception e) {
Slog.e(TAG, "Exception in user monitor.", e);
}
}
}
/**
* Performs lazy restore from the staged data.
*
* <p>This is invoked by the package monitor on the package added callback.
*/
private void doLazyRestoreLocked(String packageName, int userId) {
if (DEBUG) {
Slog.d(TAG, "doLazyRestore package=" + packageName + " user=" + userId);
}
// Check if the package is installed indeed
if (!isPackageInstalledForUser(packageName, userId)) {
Slog.e(TAG, packageName + " not installed for user " + userId
+ ". Could not restore locales from stage data");
return;
}
StagedData stagedData = mStagedData.get(userId);
for (String pkgName : stagedData.mPackageStates.keySet()) {
String languageTags = stagedData.mPackageStates.get(pkgName);
if (pkgName.equals(packageName)) {
checkExistingLocalesAndApplyRestore(pkgName, languageTags, userId);
// Remove the restored entry from the staged data list.
stagedData.mPackageStates.remove(pkgName);
// Remove the stage data entry for user if there are no more packages to restore.
if (stagedData.mPackageStates.isEmpty()) {
mStagedData.remove(userId);
}
// No need to loop further after restoring locales because the staged data will
// contain at most one entry for the newly added package.
break;
}
}
}
}