Move the somewhat expensive calculation of the "Unused apps" count to the background thread Initially, the "Unused apps" preference is unavailable. When the bg work finishes and we see we have a non-zero number of unused apps, we display the preference and update the summary text. Bug: 187996287 Test: atest HibernatedAppsPreferenceControllerTest Test: measure latency of displaying preferences w/ custom trace points Change-Id: Idb0d836fd8f4bcdd2605a7d59703a7ed53bcd6d4
193 lines
7.4 KiB
Java
193 lines
7.4 KiB
Java
/*
|
|
* 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.settings.applications;
|
|
|
|
import static android.app.usage.UsageStatsManager.INTERVAL_MONTHLY;
|
|
import static android.provider.DeviceConfig.NAMESPACE_APP_HIBERNATION;
|
|
|
|
import static com.android.settings.Utils.PROPERTY_APP_HIBERNATION_ENABLED;
|
|
|
|
import android.app.usage.UsageStats;
|
|
import android.app.usage.UsageStatsManager;
|
|
import android.apphibernation.AppHibernationManager;
|
|
import android.content.Context;
|
|
import android.content.pm.PackageInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.provider.DeviceConfig;
|
|
import android.util.ArrayMap;
|
|
|
|
import androidx.annotation.NonNull;
|
|
import androidx.annotation.WorkerThread;
|
|
import androidx.lifecycle.Lifecycle;
|
|
import androidx.lifecycle.LifecycleObserver;
|
|
import androidx.lifecycle.OnLifecycleEvent;
|
|
import androidx.preference.Preference;
|
|
import androidx.preference.PreferenceScreen;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.core.BasePreferenceController;
|
|
|
|
import com.google.common.annotations.VisibleForTesting;
|
|
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.concurrent.Executor;
|
|
import java.util.concurrent.Executors;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
/**
|
|
* A preference controller handling the logic for updating summary of hibernated apps.
|
|
*/
|
|
public final class HibernatedAppsPreferenceController extends BasePreferenceController
|
|
implements LifecycleObserver {
|
|
private static final String TAG = "HibernatedAppsPrefController";
|
|
private static final String PROPERTY_HIBERNATION_UNUSED_THRESHOLD_MILLIS =
|
|
"auto_revoke_unused_threshold_millis2";
|
|
private static final long DEFAULT_UNUSED_THRESHOLD_MS = TimeUnit.DAYS.toMillis(90);
|
|
private PreferenceScreen mScreen;
|
|
private int mUnusedCount = 0;
|
|
private boolean mLoadingUnusedApps;
|
|
private final Executor mBackgroundExecutor;
|
|
private final Executor mMainExecutor;
|
|
|
|
public HibernatedAppsPreferenceController(Context context, String preferenceKey) {
|
|
this(context, preferenceKey, Executors.newSingleThreadExecutor(),
|
|
context.getMainExecutor());
|
|
}
|
|
|
|
@VisibleForTesting
|
|
HibernatedAppsPreferenceController(Context context, String preferenceKey,
|
|
Executor bgExecutor, Executor mainExecutor) {
|
|
super(context, preferenceKey);
|
|
mBackgroundExecutor = bgExecutor;
|
|
mMainExecutor = mainExecutor;
|
|
}
|
|
|
|
@Override
|
|
public int getAvailabilityStatus() {
|
|
return isHibernationEnabled() && mUnusedCount > 0
|
|
? AVAILABLE : CONDITIONALLY_UNAVAILABLE;
|
|
}
|
|
|
|
@Override
|
|
public CharSequence getSummary() {
|
|
return mContext.getResources().getQuantityString(
|
|
R.plurals.unused_apps_summary, mUnusedCount, mUnusedCount);
|
|
}
|
|
|
|
@Override
|
|
public void displayPreference(PreferenceScreen screen) {
|
|
super.displayPreference(screen);
|
|
mScreen = screen;
|
|
}
|
|
|
|
/**
|
|
* On lifecycle resume event.
|
|
*/
|
|
@OnLifecycleEvent(Lifecycle.Event.ON_RESUME)
|
|
public void onResume() {
|
|
updatePreference();
|
|
}
|
|
|
|
private void updatePreference() {
|
|
if (mScreen == null) {
|
|
return;
|
|
}
|
|
if (!mLoadingUnusedApps) {
|
|
loadUnusedCount(unusedCount -> {
|
|
mUnusedCount = unusedCount;
|
|
mLoadingUnusedApps = false;
|
|
mMainExecutor.execute(() -> {
|
|
super.displayPreference(mScreen);
|
|
Preference pref = mScreen.findPreference(mPreferenceKey);
|
|
refreshSummary(pref);
|
|
});
|
|
});
|
|
mLoadingUnusedApps = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Asynchronously load the count of unused apps.
|
|
*
|
|
* @param callback callback to call when the number of unused apps is calculated
|
|
*/
|
|
private void loadUnusedCount(@NonNull UnusedCountLoadedCallback callback) {
|
|
mBackgroundExecutor.execute(() -> {
|
|
final int unusedCount = getUnusedCount();
|
|
callback.onUnusedCountLoaded(unusedCount);
|
|
});
|
|
}
|
|
|
|
@WorkerThread
|
|
private int getUnusedCount() {
|
|
// TODO(b/187465752): Find a way to export this logic from PermissionController module
|
|
final PackageManager pm = mContext.getPackageManager();
|
|
final AppHibernationManager ahm = mContext.getSystemService(AppHibernationManager.class);
|
|
final List<String> hibernatedPackages = ahm.getHibernatingPackagesForUser();
|
|
int numHibernated = hibernatedPackages.size();
|
|
|
|
// Also need to count packages that are auto revoked but not hibernated.
|
|
int numAutoRevoked = 0;
|
|
final UsageStatsManager usm = mContext.getSystemService(UsageStatsManager.class);
|
|
final long now = System.currentTimeMillis();
|
|
final long unusedThreshold = DeviceConfig.getLong(DeviceConfig.NAMESPACE_PERMISSIONS,
|
|
PROPERTY_HIBERNATION_UNUSED_THRESHOLD_MILLIS, DEFAULT_UNUSED_THRESHOLD_MS);
|
|
final List<UsageStats> usageStatsList = usm.queryUsageStats(INTERVAL_MONTHLY,
|
|
now - unusedThreshold, now);
|
|
final Map<String, UsageStats> recentlyUsedPackages = new ArrayMap<>();
|
|
for (UsageStats us : usageStatsList) {
|
|
recentlyUsedPackages.put(us.mPackageName, us);
|
|
}
|
|
final List<PackageInfo> packages = pm.getInstalledPackages(
|
|
PackageManager.MATCH_DISABLED_COMPONENTS | PackageManager.GET_PERMISSIONS);
|
|
for (PackageInfo pi : packages) {
|
|
final String packageName = pi.packageName;
|
|
final UsageStats usageStats = recentlyUsedPackages.get(packageName);
|
|
// Only count packages that have not been used recently as auto-revoked permissions may
|
|
// stay revoked even after use if the user has not regranted them.
|
|
final boolean usedRecently = (usageStats != null
|
|
&& (now - usageStats.getLastTimeAnyComponentUsed() < unusedThreshold
|
|
|| now - usageStats.getLastTimeVisible() < unusedThreshold));
|
|
if (!hibernatedPackages.contains(packageName)
|
|
&& pi.requestedPermissions != null
|
|
&& !usedRecently) {
|
|
for (String perm : pi.requestedPermissions) {
|
|
if ((pm.getPermissionFlags(perm, packageName, mContext.getUser())
|
|
& PackageManager.FLAG_PERMISSION_AUTO_REVOKED) != 0) {
|
|
numAutoRevoked++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return numHibernated + numAutoRevoked;
|
|
}
|
|
|
|
private static boolean isHibernationEnabled() {
|
|
return DeviceConfig.getBoolean(
|
|
NAMESPACE_APP_HIBERNATION, PROPERTY_APP_HIBERNATION_ENABLED, false);
|
|
}
|
|
|
|
/**
|
|
* Callback for when we've determined the number of unused apps.
|
|
*/
|
|
private interface UnusedCountLoadedCallback {
|
|
void onUnusedCountLoaded(int unusedCount);
|
|
}
|
|
}
|