| /* |
| * 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<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, true); |
| } |
| |
| /** |
| * Callback for when we've determined the number of unused apps. |
| */ |
| private interface UnusedCountLoadedCallback { |
| void onUnusedCountLoaded(int unusedCount); |
| } |
| } |