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 com.android.permissioncontroller.role.ui; 18 19 import android.app.role.RoleManager; 20 import android.content.Context; 21 import android.os.UserHandle; 22 import android.util.Log; 23 24 import androidx.annotation.NonNull; 25 import androidx.annotation.Nullable; 26 import androidx.lifecycle.LiveData; 27 28 import java.util.concurrent.Executor; 29 import java.util.function.Consumer; 30 31 /** 32 * {@link LiveData} for the state of managing a role holder. 33 */ 34 public class ManageRoleHolderStateLiveData extends LiveData<Integer> { 35 36 private static final String LOG_TAG = ManageRoleHolderStateLiveData.class.getSimpleName(); 37 38 private static final boolean DEBUG = false; 39 40 public static final int STATE_IDLE = 0; 41 public static final int STATE_WORKING = 1; 42 public static final int STATE_SUCCESS = 2; 43 public static final int STATE_FAILURE = 3; 44 45 @Nullable 46 private String mLastPackageName; 47 private boolean mLastAdd; 48 private int mLastFlags; 49 private UserHandle mLastUser; 50 ManageRoleHolderStateLiveData()51 public ManageRoleHolderStateLiveData() { 52 setValue(STATE_IDLE); 53 } 54 55 /** 56 * Set whether an application is a holder of a role, and update the state accordingly. Will 57 * be no-op if the current state is not {@link #STATE_IDLE}. 58 * 59 * @param roleName the name of the role 60 * @param packageName the package name of the application 61 * @param add whether to add or remove the application as a role holder 62 * @param flags optional behavior flags 63 * @param user the user to manage the role holder for 64 * @param context the {@code Context} to retrieve system services 65 */ setRoleHolderAsUser(@onNull String roleName, @NonNull String packageName, boolean add, int flags, @NonNull UserHandle user, @NonNull Context context)66 public void setRoleHolderAsUser(@NonNull String roleName, @NonNull String packageName, 67 boolean add, int flags, @NonNull UserHandle user, @NonNull Context context) { 68 if (getValue() != STATE_IDLE) { 69 Log.e(LOG_TAG, "Already (tried) managing role holders, requested role: " + roleName 70 + ", requested package: " + packageName); 71 return; 72 } 73 if (DEBUG) { 74 Log.i(LOG_TAG, (add ? "Adding" : "Removing") + " package as role holder, role: " 75 + roleName + ", package: " + packageName); 76 } 77 mLastPackageName = packageName; 78 mLastAdd = add; 79 mLastFlags = flags; 80 mLastUser = user; 81 setValue(STATE_WORKING); 82 83 RoleManager roleManager = context.getSystemService(RoleManager.class); 84 Executor executor = context.getMainExecutor(); 85 Consumer<Boolean> callback = successful -> { 86 if (successful) { 87 if (DEBUG) { 88 Log.i(LOG_TAG, "Package " + (add ? "added" : "removed") 89 + " as role holder, role: " + roleName + ", package: " + packageName); 90 } 91 setValue(STATE_SUCCESS); 92 } else { 93 if (DEBUG) { 94 Log.i(LOG_TAG, "Failed to " + (add ? "add" : "remove") 95 + " package as role holder, role: " + roleName + ", package: " 96 + packageName); 97 } 98 setValue(STATE_FAILURE); 99 } 100 }; 101 if (add) { 102 roleManager.addRoleHolderAsUser(roleName, packageName, flags, user, executor, callback); 103 } else { 104 roleManager.removeRoleHolderAsUser(roleName, packageName, flags, user, executor, 105 callback); 106 } 107 } 108 109 /** 110 * Clear the holders of a role, and update the state accordingly. Will be no-op if the current 111 * state is not {@link #STATE_IDLE}. 112 * 113 * @param roleName the name of the role 114 * @param flags optional behavior flags 115 * @param user the user to manage the role holder for 116 * @param context the {@code Context} to retrieve system services 117 */ clearRoleHoldersAsUser(@onNull String roleName, int flags, @NonNull UserHandle user, @NonNull Context context)118 public void clearRoleHoldersAsUser(@NonNull String roleName, int flags, 119 @NonNull UserHandle user, @NonNull Context context) { 120 if (getValue() != STATE_IDLE) { 121 Log.e(LOG_TAG, "Already (tried) managing role holders, requested role: " + roleName); 122 return; 123 } 124 if (DEBUG) { 125 Log.i(LOG_TAG, "Clearing role holders, role: " + roleName); 126 } 127 mLastPackageName = null; 128 mLastAdd = false; 129 mLastFlags = flags; 130 mLastUser = user; 131 setValue(STATE_WORKING); 132 133 RoleManager roleManager = context.getSystemService(RoleManager.class); 134 Executor executor = context.getMainExecutor(); 135 Consumer<Boolean> callback = successful -> { 136 if (successful) { 137 if (DEBUG) { 138 Log.i(LOG_TAG, "Cleared role holders, role: " + roleName); 139 } 140 setValue(STATE_SUCCESS); 141 } else { 142 if (DEBUG) { 143 Log.i(LOG_TAG, "Failed to clear role holders, role: " + roleName); 144 } 145 setValue(STATE_FAILURE); 146 } 147 }; 148 roleManager.clearRoleHoldersAsUser(roleName, flags, user, executor, callback); 149 } 150 151 @Nullable getLastPackageName()152 public String getLastPackageName() { 153 return mLastPackageName; 154 } 155 isLastAdd()156 public boolean isLastAdd() { 157 return mLastAdd; 158 } 159 getLastFlags()160 public int getLastFlags() { 161 return mLastFlags; 162 } 163 getLastUser()164 public UserHandle getLastUser() { 165 return mLastUser; 166 } 167 168 /** 169 * Reset the state of this live data to {@link #STATE_IDLE}. Will be no-op if the current state 170 * is not {@link #STATE_SUCCESS} or {@link #STATE_FAILURE}. 171 */ resetState()172 public void resetState() { 173 int state = getValue(); 174 if (!(state == STATE_SUCCESS || state == STATE_FAILURE)) { 175 Log.e(LOG_TAG, "Trying to reset state when the current state is not STATE_SUCCESS or" 176 + " STATE_FAILURE"); 177 return; 178 } 179 mLastPackageName = null; 180 mLastAdd = false; 181 mLastFlags = 0; 182 mLastUser = null; 183 setValue(STATE_IDLE); 184 } 185 } 186