Merge "Add the new content protection preference option and subpage in Settings -> Security & privacy -> More security & privacy settings." into main
diff --git a/src/com/android/settings/applications/AppCounter.java b/src/com/android/settings/applications/AppCounter.java
index ce2be84..d536932 100644
--- a/src/com/android/settings/applications/AppCounter.java
+++ b/src/com/android/settings/applications/AppCounter.java
@@ -16,33 +16,49 @@
import android.content.Context;
import android.content.pm.ApplicationInfo;
+import android.content.pm.FeatureFlags;
+import android.content.pm.FeatureFlagsImpl;
import android.content.pm.PackageManager;
+import android.content.pm.PackageManager.ApplicationInfoFlags;
import android.content.pm.UserInfo;
import android.os.AsyncTask;
import android.os.UserHandle;
import android.os.UserManager;
+import androidx.annotation.NonNull;
+import androidx.annotation.VisibleForTesting;
+
import java.util.List;
public abstract class AppCounter extends AsyncTask<Void, Void, Integer> {
protected final PackageManager mPm;
protected final UserManager mUm;
+ protected final FeatureFlags mFf;
- public AppCounter(Context context, PackageManager packageManager) {
+ @VisibleForTesting
+ AppCounter(@NonNull Context context, @NonNull PackageManager packageManager,
+ @NonNull FeatureFlags featureFlags) {
mPm = packageManager;
- mUm = (UserManager) context.getSystemService(Context.USER_SERVICE);
+ mUm = context.getSystemService(UserManager.class);
+ mFf = featureFlags;
+ }
+
+ public AppCounter(@NonNull Context context, @NonNull PackageManager packageManager) {
+ this(context, packageManager, new FeatureFlagsImpl());
}
@Override
protected Integer doInBackground(Void... params) {
int count = 0;
for (UserInfo user : mUm.getProfiles(UserHandle.myUserId())) {
+ long flags = PackageManager.GET_DISABLED_COMPONENTS
+ | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS
+ | (mFf.archiving() ? PackageManager.MATCH_ARCHIVED_PACKAGES : 0)
+ | (user.isAdmin() ? PackageManager.MATCH_ANY_USER : 0);
+ ApplicationInfoFlags infoFlags = ApplicationInfoFlags.of(flags);
final List<ApplicationInfo> list =
- mPm.getInstalledApplicationsAsUser(PackageManager.GET_DISABLED_COMPONENTS
- | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS
- | (user.isAdmin() ? PackageManager.MATCH_ANY_USER : 0),
- user.id);
+ mPm.getInstalledApplicationsAsUser(infoFlags, user.id);
for (ApplicationInfo info : list) {
if (includeInCount(info)) {
count++;
@@ -62,5 +78,6 @@
}
protected abstract void onCountComplete(int num);
+
protected abstract boolean includeInCount(ApplicationInfo info);
}
diff --git a/src/com/android/settings/applications/InstalledAppCounter.java b/src/com/android/settings/applications/InstalledAppCounter.java
index aeac26e..9da4c9a 100644
--- a/src/com/android/settings/applications/InstalledAppCounter.java
+++ b/src/com/android/settings/applications/InstalledAppCounter.java
@@ -17,10 +17,15 @@
import android.content.Context;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
+import android.content.pm.FeatureFlags;
+import android.content.pm.FeatureFlagsImpl;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.os.UserHandle;
+import androidx.annotation.NonNull;
+import androidx.annotation.VisibleForTesting;
+
import java.util.List;
public abstract class InstalledAppCounter extends AppCounter {
@@ -32,9 +37,15 @@
private final int mInstallReason;
- public InstalledAppCounter(Context context, int installReason,
- PackageManager packageManager) {
- super(context, packageManager);
+ public InstalledAppCounter(@NonNull Context context, int installReason,
+ @NonNull PackageManager packageManager) {
+ this(context, installReason, packageManager, new FeatureFlagsImpl());
+ }
+
+ @VisibleForTesting
+ InstalledAppCounter(@NonNull Context context, int installReason,
+ @NonNull PackageManager packageManager, @NonNull FeatureFlags featureFlags) {
+ super(context, packageManager, featureFlags);
mInstallReason = installReason;
}
diff --git a/tests/robotests/src/com/android/settings/applications/InstalledAppCounterTest.java b/tests/robotests/src/com/android/settings/applications/InstalledAppCounterTest.java
index cd9cdd6..78cfd36 100644
--- a/tests/robotests/src/com/android/settings/applications/InstalledAppCounterTest.java
+++ b/tests/robotests/src/com/android/settings/applications/InstalledAppCounterTest.java
@@ -32,7 +32,11 @@
import android.content.Context;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
+import android.content.pm.FakeFeatureFlagsImpl;
+import android.content.pm.FeatureFlags;
+import android.content.pm.Flags;
import android.content.pm.PackageManager;
+import android.content.pm.PackageManager.ApplicationInfoFlags;
import android.content.pm.ResolveInfo;
import android.content.pm.UserInfo;
import android.os.UserHandle;
@@ -51,18 +55,20 @@
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
+import java.util.List;
import java.util.Set;
@RunWith(RobolectricTestRunner.class)
@LooperMode(LooperMode.Mode.LEGACY)
public final class InstalledAppCounterTest {
- private final String APP_1 = "app1";
- private final String APP_2 = "app2";
- private final String APP_3 = "app3";
- private final String APP_4 = "app4";
- private final String APP_5 = "app5";
- private final String APP_6 = "app6";
+ private static final String APP_1 = "app1";
+ private static final String APP_2 = "app2";
+ private static final String APP_3 = "app3";
+ private static final String APP_4 = "app4";
+ private static final String APP_5 = "app5";
+ private static final String APP_6 = "app6";
+ private static final String APP_7 = "app7";
private final int MAIN_USER_ID = 0;
private final int MANAGED_PROFILE_ID = 10;
@@ -85,11 +91,16 @@
private ApplicationInfo mApp4;
private ApplicationInfo mApp5;
private ApplicationInfo mApp6;
+ private ApplicationInfo mApp7;
+
+ private FakeFeatureFlagsImpl mFakeFeatureFlags;
@Before
public void setUp() {
MockitoAnnotations.initMocks(this);
when(mContext.getSystemService(Context.USER_SERVICE)).thenReturn(mUserManager);
+ mFakeFeatureFlags = new FakeFeatureFlagsImpl();
+ mFakeFeatureFlags.setFlag(Flags.FLAG_ARCHIVING, true);
mApp1 = buildInfo(MAIN_USER_APP_UID, APP_1,
ApplicationInfo.FLAG_UPDATED_SYSTEM_APP, 0 /* targetSdkVersion */);
@@ -103,6 +114,9 @@
0 /* targetSdkVersion */);
mApp6 = buildInfo(MANAGED_PROFILE_APP_UID, APP_6, ApplicationInfo.FLAG_SYSTEM,
0 /* targetSdkVersion */);
+ mApp7 = buildInfo(MAIN_USER_APP_UID, APP_7, 0 /* flags */,
+ 0 /* targetSdkVersion */);
+ mApp7.isArchived = true;
}
private void expectQueryIntentActivities(int userId, String packageName, boolean launchable) {
@@ -128,8 +142,14 @@
// Verify that installed packages were retrieved the current user and the user's managed
// profile only.
- verify(mPackageManager).getInstalledApplicationsAsUser(anyInt(), eq(MAIN_USER_ID));
- verify(mPackageManager).getInstalledApplicationsAsUser(anyInt(), eq(MANAGED_PROFILE_ID));
+ verify(mPackageManager)
+ .getInstalledApplicationsAsUser(
+ any(ApplicationInfoFlags.class),
+ eq(MAIN_USER_ID));
+ verify(mPackageManager)
+ .getInstalledApplicationsAsUser(
+ any(ApplicationInfoFlags.class),
+ eq(MANAGED_PROFILE_ID));
verify(mPackageManager, atLeast(0))
.queryIntentActivitiesAsUser(any(Intent.class), anyInt(), anyInt());
verifyNoMoreInteractions(mPackageManager);
@@ -179,6 +199,48 @@
testCountInstalledAppsAcrossAllUsers(true /* async */);
}
+ @Test
+ public void testCountInstalledApps_archivingDisabled() {
+ when(mUserManager.getProfiles(UserHandle.myUserId())).thenReturn(List.of(
+ new UserInfo(MAIN_USER_ID, "main", UserInfo.FLAG_ADMIN)));
+ // The user has four apps installed:
+ // * app2 is a user-installed app. It should be counted.
+ // * app7 is a user-archived app. It should not be counted.
+ when(mPackageManager.getInstalledApplicationsAsUser(
+ argThat(isApplicationInfoFlagsEqualTo(
+ ApplicationInfoFlags.of(
+ PackageManager.GET_DISABLED_COMPONENTS
+ | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS
+ | PackageManager.MATCH_ANY_USER))),
+ eq(MAIN_USER_ID))).thenReturn(Arrays.asList(mApp2));
+
+ mFakeFeatureFlags.setFlag(Flags.FLAG_ARCHIVING, false);
+ // Count the number of all apps installed, irrespective of install reason.
+ count(InstalledAppCounter.IGNORE_INSTALL_REASON, mFakeFeatureFlags);
+ assertThat(mInstalledAppCount).isEqualTo(1);
+ }
+
+ @Test
+ public void testCountInstalledApps_archivingEnabled() {
+ when(mUserManager.getProfiles(UserHandle.myUserId())).thenReturn(List.of(
+ new UserInfo(MAIN_USER_ID, "main", UserInfo.FLAG_ADMIN)));
+ // The user has four apps installed:
+ // * app2 is a user-installed app. It should be counted.
+ // * app7 is a user-archived app. It should be counted.
+ when(mPackageManager.getInstalledApplicationsAsUser(
+ argThat(isApplicationInfoFlagsEqualTo(
+ ApplicationInfoFlags.of(
+ PackageManager.GET_DISABLED_COMPONENTS
+ | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS
+ | PackageManager.MATCH_ANY_USER
+ | PackageManager.MATCH_ARCHIVED_PACKAGES))),
+ eq(MAIN_USER_ID))).thenReturn(Arrays.asList(mApp2, mApp7));
+
+ // Count the number of all apps installed, irrespective of install reason.
+ count(InstalledAppCounter.IGNORE_INSTALL_REASON, mFakeFeatureFlags);
+ assertThat(mInstalledAppCount).isEqualTo(2);
+ }
+
private void count(int installReason, boolean async) {
mInstalledAppCount = -1;
final InstalledAppCounterTestable counter = new InstalledAppCounterTestable(installReason);
@@ -191,16 +253,27 @@
}
}
+ private void count(int installReason, FeatureFlags featureFlags) {
+ mInstalledAppCount = -1;
+ final InstalledAppCounterTestable counter =
+ new InstalledAppCounterTestable(installReason, featureFlags);
+ counter.executeInForeground();
+ }
+
private void configurePackageManager() {
// The first user has four apps installed:
// * app1 is an updated system app. It should be counted.
// * app2 is a user-installed app. It should be counted.
// * app3 is a system app that provides a launcher icon. It should be counted.
// * app4 is a system app that provides no launcher icon. It should not be counted.
- when(mPackageManager.getInstalledApplicationsAsUser(PackageManager.GET_DISABLED_COMPONENTS
- | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS
- | PackageManager.MATCH_ANY_USER,
- MAIN_USER_ID)).thenReturn(Arrays.asList(mApp1, mApp2, mApp3, mApp4));
+ ApplicationInfoFlags infoFlags1 = ApplicationInfoFlags.of(
+ PackageManager.GET_DISABLED_COMPONENTS
+ | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS
+ | PackageManager.MATCH_ANY_USER);
+ when(mPackageManager.getInstalledApplicationsAsUser(
+ argThat(isApplicationInfoFlagsEqualTo(infoFlags1)),
+ eq(MAIN_USER_ID))
+ ).thenReturn(Arrays.asList(mApp1, mApp2, mApp3, mApp4));
// For system apps, InstalledAppCounter checks whether they handle the default launcher
// intent to decide whether to include them in the count of installed apps or not.
expectQueryIntentActivities(MAIN_USER_ID, APP_3, true /* launchable */);
@@ -220,9 +293,12 @@
// The second user has two apps installed:
// * app5 is a user-installed app. It should be counted.
// * app6 is a system app that provides a launcher icon. It should be counted.
- when(mPackageManager.getInstalledApplicationsAsUser(PackageManager.GET_DISABLED_COMPONENTS
- | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS,MANAGED_PROFILE_ID))
- .thenReturn(Arrays.asList(mApp5, mApp6));
+ ApplicationInfoFlags infoFlags2 = ApplicationInfoFlags.of(
+ PackageManager.GET_DISABLED_COMPONENTS
+ | PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS);
+ when(mPackageManager.getInstalledApplicationsAsUser(
+ argThat(isApplicationInfoFlagsEqualTo(infoFlags2)), eq(MANAGED_PROFILE_ID))
+ ).thenReturn(Arrays.asList(mApp5, mApp6));
expectQueryIntentActivities(MANAGED_PROFILE_ID, APP_6, true /* launchable */);
// app5 is installed by enterprise policy.
@@ -238,6 +314,10 @@
super(mContext, installReason, mPackageManager);
}
+ private InstalledAppCounterTestable(int installReason, FeatureFlags featureFlags) {
+ super(mContext, installReason, mPackageManager, featureFlags);
+ }
+
@Override
protected void onCountComplete(int num) {
mInstalledAppCount = num;
@@ -263,4 +343,14 @@
return true;
};
}
+
+ private ArgumentMatcher<ApplicationInfoFlags> isApplicationInfoFlagsEqualTo(
+ ApplicationInfoFlags infoFlags) {
+ return flags -> {
+ if (flags == null) {
+ return false;
+ }
+ return flags.getValue() == infoFlags.getValue();
+ };
+ }
}