1 /*
2  * Copyright (C) 2022 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.server.healthconnect.permission;
18 
19 import android.annotation.NonNull;
20 import android.content.Context;
21 import android.content.Intent;
22 import android.content.pm.PackageManager;
23 import android.content.pm.ResolveInfo;
24 import android.health.connect.Constants;
25 import android.health.connect.HealthConnectManager;
26 import android.os.UserHandle;
27 import android.os.UserManager;
28 import android.util.ArraySet;
29 import android.util.Log;
30 
31 import com.android.internal.annotations.GuardedBy;
32 
33 import java.util.HashMap;
34 import java.util.List;
35 import java.util.Map;
36 import java.util.Set;
37 
38 /**
39  * Tracks apps which support {@link android.content.Intent#ACTION_VIEW_PERMISSION_USAGE} with {@link
40  * HealthConnectManager#CATEGORY_HEALTH_PERMISSIONS}.
41  *
42  * @hide
43  */
44 public class HealthPermissionIntentAppsTracker {
45     private static final String TAG = "HealthPermIntentTracker";
46     private static final Intent HEALTH_PERMISSIONS_USAGE_INTENT = getHealthPermissionsUsageIntent();
47 
48     private final PackageManager mPackageManager;
49     private final Object mLock = new Object();
50 
51     @GuardedBy("mLock")
52     private Map<UserHandle, Set<String>> mUserToHealthPackageNamesMap;
53 
HealthPermissionIntentAppsTracker(@onNull Context context)54     public HealthPermissionIntentAppsTracker(@NonNull Context context) {
55         mPackageManager = context.getPackageManager();
56         initPerUserMapping(context);
57     }
58 
59     /**
60      * Checks if the given app supports {@link android.content.Intent#ACTION_VIEW_PERMISSION_USAGE}
61      * with {@link HealthConnectManager#CATEGORY_HEALTH_PERMISSIONS}
62      *
63      * @param packageName: name of the package to check
64      * @param userHandle: the user to query
65      */
supportsPermissionUsageIntent( @onNull String packageName, @NonNull UserHandle userHandle)66     boolean supportsPermissionUsageIntent(
67             @NonNull String packageName, @NonNull UserHandle userHandle) {
68         // Consider readWrite lock if this is performance bottleneck.
69         synchronized (mLock) {
70             if (!mUserToHealthPackageNamesMap.containsKey(userHandle)) {
71                 Log.w(
72                         TAG,
73                         "Requested user handle: "
74                                 + userHandle.toString()
75                                 + " is not present in the state.");
76                 return false;
77             }
78 
79             if (!mUserToHealthPackageNamesMap.get(userHandle).contains(packageName)) {
80                 updateStateAndGetIfIntentWasRemoved(packageName, userHandle);
81             }
82 
83             return mUserToHealthPackageNamesMap.get(userHandle).contains(packageName);
84         }
85     }
86 
87     /**
88      * Updates package state if needed, returns whether activity for {@link
89      * android.content.Intent#ACTION_VIEW_PERMISSION_USAGE} with {@link
90      * HealthConnectManager#CATEGORY_HEALTH_PERMISSIONS} support has been disabled/removed.
91      */
updateStateAndGetIfIntentWasRemoved( @onNull String packageNameToUpdate, @NonNull UserHandle userHandle)92     boolean updateStateAndGetIfIntentWasRemoved(
93             @NonNull String packageNameToUpdate, @NonNull UserHandle userHandle) {
94         synchronized (mLock) {
95             if (!mUserToHealthPackageNamesMap.containsKey(userHandle)) {
96                 mUserToHealthPackageNamesMap.put(userHandle, new ArraySet<>());
97             }
98         }
99 
100         Intent permissionPackageUsageIntent = getHealthPermissionsUsageIntent();
101         permissionPackageUsageIntent.setPackage(packageNameToUpdate);
102         boolean removedIntent = false;
103         if (!mPackageManager
104                 .queryIntentActivitiesAsUser(
105                         permissionPackageUsageIntent,
106                         PackageManager.ResolveInfoFlags.of(PackageManager.MATCH_ALL),
107                         userHandle)
108                 .isEmpty()) {
109             synchronized (mLock) {
110                 mUserToHealthPackageNamesMap.get(userHandle).add(packageNameToUpdate);
111             }
112         } else {
113             synchronized (mLock) {
114                 removedIntent =
115                         mUserToHealthPackageNamesMap.get(userHandle).remove(packageNameToUpdate);
116             }
117         }
118         logStateIfDebugMode(userHandle);
119         return removedIntent;
120     }
121 
getHealthPermissionsUsageIntent()122     private static Intent getHealthPermissionsUsageIntent() {
123         Intent healthIntent = new Intent(Intent.ACTION_VIEW_PERMISSION_USAGE);
124         healthIntent.addCategory(HealthConnectManager.CATEGORY_HEALTH_PERMISSIONS);
125         return healthIntent;
126     }
127 
initPerUserMapping(Context context)128     private void initPerUserMapping(Context context) {
129         synchronized (mLock) {
130             mUserToHealthPackageNamesMap = new HashMap<>();
131         }
132         List<UserHandle> userHandles =
133                 context.getSystemService(UserManager.class)
134                         .getUserHandles(/* excludeDying= */ true);
135         for (UserHandle userHandle : userHandles) {
136             initPackageSetForUser(userHandle);
137         }
138     }
139 
140     /** Update list of health apps for given user. */
initPackageSetForUser(@onNull UserHandle userHandle)141     private void initPackageSetForUser(@NonNull UserHandle userHandle) {
142         List<ResolveInfo> healthAppInfos = getHealthIntentSupportiveAppsForUser(userHandle);
143         Set<String> healthApps = new ArraySet<String>(healthAppInfos.size());
144         for (ResolveInfo info : healthAppInfos) {
145             String packageName = extractPackageName(info);
146             if (packageName != null) {
147                 healthApps.add(packageName);
148             }
149         }
150         synchronized (mLock) {
151             mUserToHealthPackageNamesMap.put(userHandle, healthApps);
152         }
153         logStateIfDebugMode(userHandle);
154     }
155 
getHealthIntentSupportiveAppsForUser(@onNull UserHandle userHandle)156     private List<ResolveInfo> getHealthIntentSupportiveAppsForUser(@NonNull UserHandle userHandle) {
157         return mPackageManager.queryIntentActivitiesAsUser(
158                 HEALTH_PERMISSIONS_USAGE_INTENT,
159                 PackageManager.ResolveInfoFlags.of(PackageManager.MATCH_ALL),
160                 userHandle);
161     }
162 
163     @SuppressWarnings("NullAway") // TODO(b/317029272): fix this suppression
extractPackageName(ResolveInfo info)164     private String extractPackageName(ResolveInfo info) {
165         if (info == null
166                 || info.activityInfo == null
167                 || info.activityInfo.applicationInfo == null) {
168             Log.w(TAG, "Can't fetch application info from resolve info.");
169             return null;
170         }
171         return info.activityInfo.applicationInfo.packageName;
172     }
173 
logStateIfDebugMode(@onNull UserHandle userHandle)174     private void logStateIfDebugMode(@NonNull UserHandle userHandle) {
175         if (Constants.DEBUG) {
176             Log.d(TAG, "State for user: " + userHandle.getIdentifier());
177             synchronized (mLock) {
178                 Log.d(TAG, mUserToHealthPackageNamesMap.toString());
179             }
180         }
181     }
182 }
183