/* * 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 boolean mLoadedUnusedCount; 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() ? AVAILABLE : CONDITIONALLY_UNAVAILABLE; } @Override public CharSequence getSummary() { return mLoadedUnusedCount ? mContext.getResources().getQuantityString( R.plurals.unused_apps_summary, mUnusedCount, mUnusedCount) : mContext.getResources().getString(R.string.summary_placeholder); } @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; mLoadedUnusedCount = true; mMainExecutor.execute(() -> { 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 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 usageStatsList = usm.queryUsageStats(INTERVAL_MONTHLY, now - unusedThreshold, now); final Map recentlyUsedPackages = new ArrayMap<>(); for (UsageStats us : usageStatsList) { recentlyUsedPackages.put(us.mPackageName, us); } final List 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, true); } /** * Callback for when we've determined the number of unused apps. */ private interface UnusedCountLoadedCallback { void onUnusedCountLoaded(int unusedCount); } }