1 /*
2  * Copyright (C) 2018 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 android.app.role;
18 
19 import android.Manifest;
20 import android.annotation.CallbackExecutor;
21 import android.annotation.IntDef;
22 import android.annotation.NonNull;
23 import android.annotation.Nullable;
24 import android.annotation.RequiresPermission;
25 import android.annotation.SystemApi;
26 import android.annotation.SystemService;
27 import android.annotation.TestApi;
28 import android.annotation.UserIdInt;
29 import android.content.Context;
30 import android.content.Intent;
31 import android.os.Binder;
32 import android.os.Process;
33 import android.os.RemoteCallback;
34 import android.os.RemoteException;
35 import android.os.ServiceManager;
36 import android.os.UserHandle;
37 import android.util.ArrayMap;
38 import android.util.SparseArray;
39 
40 import com.android.internal.annotations.GuardedBy;
41 import com.android.internal.util.Preconditions;
42 import com.android.internal.util.function.pooled.PooledLambda;
43 
44 import java.util.List;
45 import java.util.Objects;
46 import java.util.concurrent.Executor;
47 import java.util.function.Consumer;
48 
49 /**
50  * This class provides information about and manages roles.
51  * <p>
52  * A role is a unique name within the system associated with certain privileges. The list of
53  * available roles might change with a system app update, so apps should not make assumption about
54  * the availability of roles. Instead, they should always query if the role is available using
55  * {@link #isRoleAvailable(String)} before trying to do anything with it. Some predefined role names
56  * are available as constants in this class, and a list of possibly available roles can be found in
57  * the <a href="{@docRoot}reference/androidx/core/role/package-summary.html">AndroidX Role
58  * library</a>.
59  * <p>
60  * There can be multiple applications qualifying for a role, but only a subset of them can become
61  * role holders. To qualify for a role, an application must meet certain requirements, including
62  * defining certain components in its manifest. These requirements can be found in the AndroidX
63  * Libraries. Then the application will need user consent to become a role holder, which can be
64  * requested using {@link android.app.Activity#startActivityForResult(Intent, int)} with the
65  * {@code Intent} obtained from {@link #createRequestRoleIntent(String)}.
66  * <p>
67  * Upon becoming a role holder, the application may be granted certain privileges that are role
68  * specific. When the application loses its role, these privileges will also be revoked.
69  */
70 @SystemService(Context.ROLE_SERVICE)
71 public final class RoleManager {
72 
73     private static final String LOG_TAG = RoleManager.class.getSimpleName();
74 
75     /**
76      * The name of the assistant app role.
77      *
78      * @see android.service.voice.VoiceInteractionService
79      */
80     public static final String ROLE_ASSISTANT = "android.app.role.ASSISTANT";
81 
82     /**
83      * The name of the browser role.
84      *
85      * @see Intent#CATEGORY_APP_BROWSER
86      */
87     public static final String ROLE_BROWSER = "android.app.role.BROWSER";
88 
89     /**
90      * The name of the dialer role.
91      *
92      * @see Intent#ACTION_DIAL
93      * @see android.telecom.InCallService
94      */
95     public static final String ROLE_DIALER = "android.app.role.DIALER";
96 
97     /**
98      * The name of the SMS role.
99      *
100      * @see Intent#CATEGORY_APP_MESSAGING
101      */
102     public static final String ROLE_SMS = "android.app.role.SMS";
103 
104     /**
105      * The name of the emergency role
106      */
107     public static final String ROLE_EMERGENCY = "android.app.role.EMERGENCY";
108 
109     /**
110      * The name of the home role.
111      *
112      * @see Intent#CATEGORY_HOME
113      */
114     public static final String ROLE_HOME = "android.app.role.HOME";
115 
116     /**
117      * The name of the call redirection role.
118      * <p>
119      * A call redirection app provides a means to re-write the phone number for an outgoing call to
120      * place the call through a call redirection service.
121      *
122      * @see android.telecom.CallRedirectionService
123      */
124     public static final String ROLE_CALL_REDIRECTION = "android.app.role.CALL_REDIRECTION";
125 
126     /**
127      * The name of the call screening and caller id role.
128      *
129      * @see android.telecom.CallScreeningService
130      */
131     public static final String ROLE_CALL_SCREENING = "android.app.role.CALL_SCREENING";
132 
133     /**
134      * @hide
135      */
136     @IntDef(flag = true, value = { MANAGE_HOLDERS_FLAG_DONT_KILL_APP })
137     public @interface ManageHoldersFlags {}
138 
139     /**
140      * Flag parameter for {@link #addRoleHolderAsUser}, {@link #removeRoleHolderAsUser} and
141      * {@link #clearRoleHoldersAsUser} to indicate that apps should not be killed when changing
142      * their role holder status.
143      *
144      * @hide
145      */
146     @SystemApi
147     @TestApi
148     public static final int MANAGE_HOLDERS_FLAG_DONT_KILL_APP = 1;
149 
150     /**
151      * The action used to request user approval of a role for an application.
152      *
153      * @hide
154      */
155     public static final String ACTION_REQUEST_ROLE = "android.app.role.action.REQUEST_ROLE";
156 
157     /**
158      * The permission required to manage records of role holders in {@link RoleManager} directly.
159      *
160      * @hide
161      */
162     public static final String PERMISSION_MANAGE_ROLES_FROM_CONTROLLER =
163             "com.android.permissioncontroller.permission.MANAGE_ROLES_FROM_CONTROLLER";
164 
165     @NonNull
166     private final Context mContext;
167 
168     @NonNull
169     private final IRoleManager mService;
170 
171     @GuardedBy("mListenersLock")
172     @NonNull
173     private final SparseArray<ArrayMap<OnRoleHoldersChangedListener,
174             OnRoleHoldersChangedListenerDelegate>> mListeners = new SparseArray<>();
175     @NonNull
176     private final Object mListenersLock = new Object();
177 
178     /**
179      * @hide
180      */
RoleManager(@onNull Context context)181     public RoleManager(@NonNull Context context) throws ServiceManager.ServiceNotFoundException {
182         mContext = context;
183         mService = IRoleManager.Stub.asInterface(ServiceManager.getServiceOrThrow(
184                 Context.ROLE_SERVICE));
185     }
186 
187     /**
188      * Returns an {@code Intent} suitable for passing to
189      * {@link android.app.Activity#startActivityForResult(Intent, int)} which prompts the user to
190      * grant a role to this application.
191      * <p>
192      * If the role is granted, the {@code resultCode} will be
193      * {@link android.app.Activity#RESULT_OK}, otherwise it will be
194      * {@link android.app.Activity#RESULT_CANCELED}.
195      *
196      * @param roleName the name of requested role
197      *
198      * @return the {@code Intent} to prompt user to grant the role
199      */
200     @NonNull
createRequestRoleIntent(@onNull String roleName)201     public Intent createRequestRoleIntent(@NonNull String roleName) {
202         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
203         Intent intent = new Intent(ACTION_REQUEST_ROLE);
204         intent.setPackage(mContext.getPackageManager().getPermissionControllerPackageName());
205         intent.putExtra(Intent.EXTRA_ROLE_NAME, roleName);
206         return intent;
207     }
208 
209     /**
210      * Check whether a role is available in the system.
211      *
212      * @param roleName the name of role to checking for
213      *
214      * @return whether the role is available in the system
215      */
isRoleAvailable(@onNull String roleName)216     public boolean isRoleAvailable(@NonNull String roleName) {
217         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
218         try {
219             return mService.isRoleAvailable(roleName);
220         } catch (RemoteException e) {
221             throw e.rethrowFromSystemServer();
222         }
223     }
224 
225     /**
226      * Check whether the calling application is holding a particular role.
227      *
228      * @param roleName the name of the role to check for
229      *
230      * @return whether the calling application is holding the role
231      */
isRoleHeld(@onNull String roleName)232     public boolean isRoleHeld(@NonNull String roleName) {
233         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
234         try {
235             return mService.isRoleHeld(roleName, mContext.getPackageName());
236         } catch (RemoteException e) {
237             throw e.rethrowFromSystemServer();
238         }
239     }
240 
241     /**
242      * Get package names of the applications holding the role.
243      * <p>
244      * <strong>Note:</strong> Using this API requires holding
245      * {@code android.permission.MANAGE_ROLE_HOLDERS}.
246      *
247      * @param roleName the name of the role to get the role holder for
248      *
249      * @return a list of package names of the role holders, or an empty list if none.
250      *
251      * @see #getRoleHoldersAsUser(String, UserHandle)
252      *
253      * @hide
254      */
255     @NonNull
256     @RequiresPermission(Manifest.permission.MANAGE_ROLE_HOLDERS)
257     @SystemApi
258     @TestApi
getRoleHolders(@onNull String roleName)259     public List<String> getRoleHolders(@NonNull String roleName) {
260         return getRoleHoldersAsUser(roleName, Process.myUserHandle());
261     }
262 
263     /**
264      * Get package names of the applications holding the role.
265      * <p>
266      * <strong>Note:</strong> Using this API requires holding
267      * {@code android.permission.MANAGE_ROLE_HOLDERS} and if the user id is not the current user
268      * {@code android.permission.INTERACT_ACROSS_USERS_FULL}.
269      *
270      * @param roleName the name of the role to get the role holder for
271      * @param user the user to get the role holder for
272      *
273      * @return a list of package names of the role holders, or an empty list if none.
274      *
275      * @see #addRoleHolderAsUser(String, String, int, UserHandle, Executor, Consumer)
276      * @see #removeRoleHolderAsUser(String, String, int, UserHandle, Executor, Consumer)
277      * @see #clearRoleHoldersAsUser(String, int, UserHandle, Executor, Consumer)
278      *
279      * @hide
280      */
281     @NonNull
282     @RequiresPermission(Manifest.permission.MANAGE_ROLE_HOLDERS)
283     @SystemApi
284     @TestApi
getRoleHoldersAsUser(@onNull String roleName, @NonNull UserHandle user)285     public List<String> getRoleHoldersAsUser(@NonNull String roleName, @NonNull UserHandle user) {
286         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
287         Objects.requireNonNull(user, "user cannot be null");
288         try {
289             return mService.getRoleHoldersAsUser(roleName, user.getIdentifier());
290         } catch (RemoteException e) {
291             throw e.rethrowFromSystemServer();
292         }
293     }
294 
295     /**
296      * Add a specific application to the holders of a role. If the role is exclusive, the previous
297      * holder will be replaced.
298      * <p>
299      * <strong>Note:</strong> Using this API requires holding
300      * {@code android.permission.MANAGE_ROLE_HOLDERS} and if the user id is not the current user
301      * {@code android.permission.INTERACT_ACROSS_USERS_FULL}.
302      *
303      * @param roleName the name of the role to add the role holder for
304      * @param packageName the package name of the application to add to the role holders
305      * @param flags optional behavior flags
306      * @param user the user to add the role holder for
307      * @param executor the {@code Executor} to run the callback on.
308      * @param callback the callback for whether this call is successful
309      *
310      * @see #getRoleHoldersAsUser(String, UserHandle)
311      * @see #removeRoleHolderAsUser(String, String, int, UserHandle, Executor, Consumer)
312      * @see #clearRoleHoldersAsUser(String, int, UserHandle, Executor, Consumer)
313      *
314      * @hide
315      */
316     @RequiresPermission(Manifest.permission.MANAGE_ROLE_HOLDERS)
317     @SystemApi
318     @TestApi
addRoleHolderAsUser(@onNull String roleName, @NonNull String packageName, @ManageHoldersFlags int flags, @NonNull UserHandle user, @CallbackExecutor @NonNull Executor executor, @NonNull Consumer<Boolean> callback)319     public void addRoleHolderAsUser(@NonNull String roleName, @NonNull String packageName,
320             @ManageHoldersFlags int flags, @NonNull UserHandle user,
321             @CallbackExecutor @NonNull Executor executor, @NonNull Consumer<Boolean> callback) {
322         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
323         Preconditions.checkStringNotEmpty(packageName, "packageName cannot be null or empty");
324         Objects.requireNonNull(user, "user cannot be null");
325         Objects.requireNonNull(executor, "executor cannot be null");
326         Objects.requireNonNull(callback, "callback cannot be null");
327         try {
328             mService.addRoleHolderAsUser(roleName, packageName, flags, user.getIdentifier(),
329                     createRemoteCallback(executor, callback));
330         } catch (RemoteException e) {
331             throw e.rethrowFromSystemServer();
332         }
333     }
334 
335     /**
336      * Remove a specific application from the holders of a role.
337      * <p>
338      * <strong>Note:</strong> Using this API requires holding
339      * {@code android.permission.MANAGE_ROLE_HOLDERS} and if the user id is not the current user
340      * {@code android.permission.INTERACT_ACROSS_USERS_FULL}.
341      *
342      * @param roleName the name of the role to remove the role holder for
343      * @param packageName the package name of the application to remove from the role holders
344      * @param flags optional behavior flags
345      * @param user the user to remove the role holder for
346      * @param executor the {@code Executor} to run the callback on.
347      * @param callback the callback for whether this call is successful
348      *
349      * @see #getRoleHoldersAsUser(String, UserHandle)
350      * @see #addRoleHolderAsUser(String, String, int, UserHandle, Executor, Consumer)
351      * @see #clearRoleHoldersAsUser(String, int, UserHandle, Executor, Consumer)
352      *
353      * @hide
354      */
355     @RequiresPermission(Manifest.permission.MANAGE_ROLE_HOLDERS)
356     @SystemApi
357     @TestApi
removeRoleHolderAsUser(@onNull String roleName, @NonNull String packageName, @ManageHoldersFlags int flags, @NonNull UserHandle user, @CallbackExecutor @NonNull Executor executor, @NonNull Consumer<Boolean> callback)358     public void removeRoleHolderAsUser(@NonNull String roleName, @NonNull String packageName,
359             @ManageHoldersFlags int flags, @NonNull UserHandle user,
360             @CallbackExecutor @NonNull Executor executor, @NonNull Consumer<Boolean> callback) {
361         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
362         Preconditions.checkStringNotEmpty(packageName, "packageName cannot be null or empty");
363         Objects.requireNonNull(user, "user cannot be null");
364         Objects.requireNonNull(executor, "executor cannot be null");
365         Objects.requireNonNull(callback, "callback cannot be null");
366         try {
367             mService.removeRoleHolderAsUser(roleName, packageName, flags, user.getIdentifier(),
368                     createRemoteCallback(executor, callback));
369         } catch (RemoteException e) {
370             throw e.rethrowFromSystemServer();
371         }
372     }
373 
374     /**
375      * Remove all holders of a role.
376      * <p>
377      * <strong>Note:</strong> Using this API requires holding
378      * {@code android.permission.MANAGE_ROLE_HOLDERS} and if the user id is not the current user
379      * {@code android.permission.INTERACT_ACROSS_USERS_FULL}.
380      *
381      * @param roleName the name of the role to remove role holders for
382      * @param flags optional behavior flags
383      * @param user the user to remove role holders for
384      * @param executor the {@code Executor} to run the callback on.
385      * @param callback the callback for whether this call is successful
386      *
387      * @see #getRoleHoldersAsUser(String, UserHandle)
388      * @see #addRoleHolderAsUser(String, String, int, UserHandle, Executor, Consumer)
389      * @see #removeRoleHolderAsUser(String, String, int, UserHandle, Executor, Consumer)
390      *
391      * @hide
392      */
393     @RequiresPermission(Manifest.permission.MANAGE_ROLE_HOLDERS)
394     @SystemApi
395     @TestApi
clearRoleHoldersAsUser(@onNull String roleName, @ManageHoldersFlags int flags, @NonNull UserHandle user, @CallbackExecutor @NonNull Executor executor, @NonNull Consumer<Boolean> callback)396     public void clearRoleHoldersAsUser(@NonNull String roleName, @ManageHoldersFlags int flags,
397             @NonNull UserHandle user, @CallbackExecutor @NonNull Executor executor,
398             @NonNull Consumer<Boolean> callback) {
399         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
400         Objects.requireNonNull(user, "user cannot be null");
401         Objects.requireNonNull(executor, "executor cannot be null");
402         Objects.requireNonNull(callback, "callback cannot be null");
403         try {
404             mService.clearRoleHoldersAsUser(roleName, flags, user.getIdentifier(),
405                     createRemoteCallback(executor, callback));
406         } catch (RemoteException e) {
407             throw e.rethrowFromSystemServer();
408         }
409     }
410 
411     @NonNull
createRemoteCallback(@onNull Executor executor, @NonNull Consumer<Boolean> callback)412     private static RemoteCallback createRemoteCallback(@NonNull Executor executor,
413             @NonNull Consumer<Boolean> callback) {
414         return new RemoteCallback(result -> executor.execute(() -> {
415             boolean successful = result != null;
416             long token = Binder.clearCallingIdentity();
417             try {
418                 callback.accept(successful);
419             } finally {
420                 Binder.restoreCallingIdentity(token);
421             }
422         }));
423     }
424 
425     /**
426      * Add a listener to observe role holder changes
427      * <p>
428      * <strong>Note:</strong> Using this API requires holding
429      * {@code android.permission.OBSERVE_ROLE_HOLDERS} and if the user id is not the current user
430      * {@code android.permission.INTERACT_ACROSS_USERS_FULL}.
431      *
432      * @param executor the {@code Executor} to call the listener on.
433      * @param listener the listener to be added
434      * @param user the user to add the listener for
435      *
436      * @see #removeOnRoleHoldersChangedListenerAsUser(OnRoleHoldersChangedListener, UserHandle)
437      *
438      * @hide
439      */
440     @RequiresPermission(Manifest.permission.OBSERVE_ROLE_HOLDERS)
441     @SystemApi
442     @TestApi
addOnRoleHoldersChangedListenerAsUser(@allbackExecutor @onNull Executor executor, @NonNull OnRoleHoldersChangedListener listener, @NonNull UserHandle user)443     public void addOnRoleHoldersChangedListenerAsUser(@CallbackExecutor @NonNull Executor executor,
444             @NonNull OnRoleHoldersChangedListener listener, @NonNull UserHandle user) {
445         Objects.requireNonNull(executor, "executor cannot be null");
446         Objects.requireNonNull(listener, "listener cannot be null");
447         Objects.requireNonNull(user, "user cannot be null");
448         int userId = user.getIdentifier();
449         synchronized (mListenersLock) {
450             ArrayMap<OnRoleHoldersChangedListener, OnRoleHoldersChangedListenerDelegate> listeners =
451                     mListeners.get(userId);
452             if (listeners == null) {
453                 listeners = new ArrayMap<>();
454                 mListeners.put(userId, listeners);
455             } else {
456                 if (listeners.containsKey(listener)) {
457                     return;
458                 }
459             }
460             OnRoleHoldersChangedListenerDelegate listenerDelegate =
461                     new OnRoleHoldersChangedListenerDelegate(executor, listener);
462             try {
463                 mService.addOnRoleHoldersChangedListenerAsUser(listenerDelegate, userId);
464             } catch (RemoteException e) {
465                 throw e.rethrowFromSystemServer();
466             }
467             listeners.put(listener, listenerDelegate);
468         }
469     }
470 
471     /**
472      * Remove a listener observing role holder changes
473      * <p>
474      * <strong>Note:</strong> Using this API requires holding
475      * {@code android.permission.OBSERVE_ROLE_HOLDERS} and if the user id is not the current user
476      * {@code android.permission.INTERACT_ACROSS_USERS_FULL}.
477      *
478      * @param listener the listener to be removed
479      * @param user the user to remove the listener for
480      *
481      * @see #addOnRoleHoldersChangedListenerAsUser(Executor, OnRoleHoldersChangedListener,
482      *                                             UserHandle)
483      *
484      * @hide
485      */
486     @RequiresPermission(Manifest.permission.OBSERVE_ROLE_HOLDERS)
487     @SystemApi
488     @TestApi
removeOnRoleHoldersChangedListenerAsUser( @onNull OnRoleHoldersChangedListener listener, @NonNull UserHandle user)489     public void removeOnRoleHoldersChangedListenerAsUser(
490             @NonNull OnRoleHoldersChangedListener listener, @NonNull UserHandle user) {
491         Objects.requireNonNull(listener, "listener cannot be null");
492         Objects.requireNonNull(user, "user cannot be null");
493         int userId = user.getIdentifier();
494         synchronized (mListenersLock) {
495             ArrayMap<OnRoleHoldersChangedListener, OnRoleHoldersChangedListenerDelegate> listeners =
496                     mListeners.get(userId);
497             if (listeners == null) {
498                 return;
499             }
500             OnRoleHoldersChangedListenerDelegate listenerDelegate = listeners.get(listener);
501             if (listenerDelegate == null) {
502                 return;
503             }
504             try {
505                 mService.removeOnRoleHoldersChangedListenerAsUser(listenerDelegate,
506                         user.getIdentifier());
507             } catch (RemoteException e) {
508                 throw e.rethrowFromSystemServer();
509             }
510             listeners.remove(listener);
511             if (listeners.isEmpty()) {
512                 mListeners.remove(userId);
513             }
514         }
515     }
516 
517     /**
518      * Set the names of all the available roles. Should only be called from
519      * {@link android.app.role.RoleControllerService}.
520      * <p>
521      * <strong>Note:</strong> Using this API requires holding
522      * {@link #PERMISSION_MANAGE_ROLES_FROM_CONTROLLER}.
523      *
524      * @param roleNames the names of all the available roles
525      *
526      * @hide
527      */
528     @RequiresPermission(PERMISSION_MANAGE_ROLES_FROM_CONTROLLER)
529     @SystemApi
530     @TestApi
setRoleNamesFromController(@onNull List<String> roleNames)531     public void setRoleNamesFromController(@NonNull List<String> roleNames) {
532         Objects.requireNonNull(roleNames, "roleNames cannot be null");
533         try {
534             mService.setRoleNamesFromController(roleNames);
535         } catch (RemoteException e) {
536             throw e.rethrowFromSystemServer();
537         }
538     }
539 
540     /**
541      * Add a specific application to the holders of a role, only modifying records inside
542      * {@link RoleManager}. Should only be called from
543      * {@link android.app.role.RoleControllerService}.
544      * <p>
545      * <strong>Note:</strong> Using this API requires holding
546      * {@link #PERMISSION_MANAGE_ROLES_FROM_CONTROLLER}.
547      *
548      * @param roleName the name of the role to add the role holder for
549      * @param packageName the package name of the application to add to the role holders
550      *
551      * @return whether the operation was successful, and will also be {@code true} if a matching
552      *         role holder is already found.
553      *
554      * @see #getRoleHolders(String)
555      * @see #removeRoleHolderFromController(String, String)
556      *
557      * @hide
558      */
559     @RequiresPermission(PERMISSION_MANAGE_ROLES_FROM_CONTROLLER)
560     @SystemApi
561     @TestApi
addRoleHolderFromController(@onNull String roleName, @NonNull String packageName)562     public boolean addRoleHolderFromController(@NonNull String roleName,
563             @NonNull String packageName) {
564         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
565         Preconditions.checkStringNotEmpty(packageName, "packageName cannot be null or empty");
566         try {
567             return mService.addRoleHolderFromController(roleName, packageName);
568         } catch (RemoteException e) {
569             throw e.rethrowFromSystemServer();
570         }
571     }
572 
573     /**
574      * Remove a specific application from the holders of a role, only modifying records inside
575      * {@link RoleManager}. Should only be called from
576      * {@link android.app.role.RoleControllerService}.
577      * <p>
578      * <strong>Note:</strong> Using this API requires holding
579      * {@link #PERMISSION_MANAGE_ROLES_FROM_CONTROLLER}.
580      *
581      * @param roleName the name of the role to remove the role holder for
582      * @param packageName the package name of the application to remove from the role holders
583      *
584      * @return whether the operation was successful, and will also be {@code true} if no matching
585      *         role holder was found to remove.
586      *
587      * @see #getRoleHolders(String)
588      * @see #addRoleHolderFromController(String, String)
589      *
590      * @hide
591      */
592     @RequiresPermission(PERMISSION_MANAGE_ROLES_FROM_CONTROLLER)
593     @SystemApi
594     @TestApi
removeRoleHolderFromController(@onNull String roleName, @NonNull String packageName)595     public boolean removeRoleHolderFromController(@NonNull String roleName,
596             @NonNull String packageName) {
597         Preconditions.checkStringNotEmpty(roleName, "roleName cannot be null or empty");
598         Preconditions.checkStringNotEmpty(packageName, "packageName cannot be null or empty");
599         try {
600             return mService.removeRoleHolderFromController(roleName, packageName);
601         } catch (RemoteException e) {
602             throw e.rethrowFromSystemServer();
603         }
604     }
605 
606     /**
607      * Returns the list of all roles that the given package is currently holding
608      *
609      * @param packageName the package name
610      * @return the list of role names
611      *
612      * @hide
613      */
614     @NonNull
615     @RequiresPermission(PERMISSION_MANAGE_ROLES_FROM_CONTROLLER)
616     @SystemApi
617     @TestApi
getHeldRolesFromController(@onNull String packageName)618     public List<String> getHeldRolesFromController(@NonNull String packageName) {
619         Preconditions.checkStringNotEmpty(packageName, "packageName cannot be null or empty");
620         try {
621             return mService.getHeldRolesFromController(packageName);
622         } catch (RemoteException e) {
623             throw e.rethrowFromSystemServer();
624         }
625     }
626 
627     /**
628      * Allows getting the role holder for {@link #ROLE_SMS} without
629      * {@link Manifest.permission#OBSERVE_ROLE_HOLDERS}, as required by
630      * {@link android.provider.Telephony.Sms#getDefaultSmsPackage(Context)}
631      *
632      * @param userId The user ID to get the default SMS package for.
633      * @return the package name of the default SMS app, or {@code null} if not configured.
634      * @hide
635      */
636     @Nullable
getDefaultSmsPackage(@serIdInt int userId)637     public String getDefaultSmsPackage(@UserIdInt int userId) {
638         try {
639             return mService.getDefaultSmsPackage(userId);
640         } catch (RemoteException e) {
641             throw e.rethrowFromSystemServer();
642         }
643     }
644 
645     private static class OnRoleHoldersChangedListenerDelegate
646             extends IOnRoleHoldersChangedListener.Stub {
647 
648         @NonNull
649         private final Executor mExecutor;
650         @NonNull
651         private final OnRoleHoldersChangedListener mListener;
652 
OnRoleHoldersChangedListenerDelegate(@onNull Executor executor, @NonNull OnRoleHoldersChangedListener listener)653         OnRoleHoldersChangedListenerDelegate(@NonNull Executor executor,
654                 @NonNull OnRoleHoldersChangedListener listener) {
655             mExecutor = executor;
656             mListener = listener;
657         }
658 
659         @Override
onRoleHoldersChanged(@onNull String roleName, @UserIdInt int userId)660         public void onRoleHoldersChanged(@NonNull String roleName, @UserIdInt int userId) {
661             long token = Binder.clearCallingIdentity();
662             try {
663                 mExecutor.execute(PooledLambda.obtainRunnable(
664                         OnRoleHoldersChangedListener::onRoleHoldersChanged, mListener, roleName,
665                         UserHandle.of(userId)));
666             } finally {
667                 Binder.restoreCallingIdentity(token);
668             }
669         }
670     }
671 }
672