1 /* 2 * Copyright (C) 2019 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.service.controls.actions; 18 19 import android.annotation.CallSuper; 20 import android.annotation.IntDef; 21 import android.annotation.NonNull; 22 import android.annotation.Nullable; 23 import android.os.Bundle; 24 import android.service.controls.Control; 25 import android.service.controls.ControlsProviderService; 26 import android.service.controls.templates.ControlTemplate; 27 import android.util.Log; 28 29 import com.android.internal.util.Preconditions; 30 31 import java.lang.annotation.Retention; 32 import java.lang.annotation.RetentionPolicy; 33 34 /** 35 * An abstract action indicating a user interaction with a {@link Control}. 36 * 37 * In some cases, an action needs to be validated by the user, using a password, PIN or simple 38 * acknowledgment. For those cases, an optional (nullable) parameter can be passed to send the user 39 * input. This <b>challenge value</b> will be requested from the user and sent as part 40 * of a {@link ControlAction} only if the service has responded to an action with one of: 41 * <ul> 42 * <li> {@link #RESPONSE_CHALLENGE_ACK} 43 * <li> {@link #RESPONSE_CHALLENGE_PIN} 44 * <li> {@link #RESPONSE_CHALLENGE_PASSPHRASE} 45 * </ul> 46 */ 47 public abstract class ControlAction { 48 49 private static final String TAG = "ControlAction"; 50 51 private static final String KEY_ACTION_TYPE = "key_action_type"; 52 private static final String KEY_TEMPLATE_ID = "key_template_id"; 53 private static final String KEY_CHALLENGE_VALUE = "key_challenge_value"; 54 55 /** 56 * @hide 57 */ 58 @Retention(RetentionPolicy.SOURCE) 59 @IntDef({ 60 TYPE_ERROR, 61 TYPE_BOOLEAN, 62 TYPE_FLOAT, 63 TYPE_MODE, 64 TYPE_COMMAND 65 }) 66 public @interface ActionType {}; 67 68 /** 69 * Object returned when there is an unparcelling error. 70 * @hide 71 */ 72 public static final @NonNull ControlAction ERROR_ACTION = new ControlAction() { 73 @Override 74 public int getActionType() { 75 return TYPE_ERROR; 76 } 77 }; 78 79 /** 80 * The identifier of the action returned by {@link #getErrorAction}. 81 */ 82 public static final @ActionType int TYPE_ERROR = -1; 83 84 /** 85 * The identifier of {@link BooleanAction}. 86 */ 87 public static final @ActionType int TYPE_BOOLEAN = 1; 88 89 /** 90 * The identifier of {@link FloatAction}. 91 */ 92 public static final @ActionType int TYPE_FLOAT = 2; 93 94 /** 95 * The identifier of {@link ModeAction}. 96 */ 97 public static final @ActionType int TYPE_MODE = 4; 98 99 /** 100 * The identifier of {@link CommandAction}. 101 */ 102 public static final @ActionType int TYPE_COMMAND = 5; 103 104 isValidResponse(@esponseResult int response)105 public static final boolean isValidResponse(@ResponseResult int response) { 106 return (response >= 0 && response < NUM_RESPONSE_TYPES); 107 } 108 private static final int NUM_RESPONSE_TYPES = 6; 109 /** 110 * @hide 111 */ 112 @Retention(RetentionPolicy.SOURCE) 113 @IntDef({ 114 RESPONSE_UNKNOWN, 115 RESPONSE_OK, 116 RESPONSE_FAIL, 117 RESPONSE_CHALLENGE_ACK, 118 RESPONSE_CHALLENGE_PIN, 119 RESPONSE_CHALLENGE_PASSPHRASE 120 }) 121 public @interface ResponseResult {}; 122 123 public static final @ResponseResult int RESPONSE_UNKNOWN = 0; 124 125 /** 126 * Response code for the {@code consumer} in 127 * {@link ControlsProviderService#performControlAction} indicating that the action has been 128 * performed. The action may still fail later and the state may not change. 129 */ 130 public static final @ResponseResult int RESPONSE_OK = 1; 131 /** 132 * Response code for the {@code consumer} in 133 * {@link ControlsProviderService#performControlAction} indicating that the action has failed. 134 */ 135 public static final @ResponseResult int RESPONSE_FAIL = 2; 136 /** 137 * Response code for the {@code consumer} in 138 * {@link ControlsProviderService#performControlAction} indicating that in order for the action 139 * to be performed, acknowledgment from the user is required. Any non-empty string returned 140 * from {@link #getChallengeValue} shall be treated as a positive acknowledgment. 141 */ 142 public static final @ResponseResult int RESPONSE_CHALLENGE_ACK = 3; 143 /** 144 * Response code for the {@code consumer} in 145 * {@link ControlsProviderService#performControlAction} indicating that in order for the action 146 * to be performed, a PIN is required. 147 */ 148 public static final @ResponseResult int RESPONSE_CHALLENGE_PIN = 4; 149 /** 150 * Response code for the {@code consumer} in 151 * {@link ControlsProviderService#performControlAction} indicating that in order for the action 152 * to be performed, an alphanumeric passphrase is required. 153 */ 154 public static final @ResponseResult int RESPONSE_CHALLENGE_PASSPHRASE = 5; 155 156 /** 157 * The action type associated with this class. 158 */ getActionType()159 public abstract @ActionType int getActionType(); 160 161 private final @NonNull String mTemplateId; 162 private final @Nullable String mChallengeValue; 163 ControlAction()164 private ControlAction() { 165 mTemplateId = ""; 166 mChallengeValue = null; 167 } 168 169 /** 170 * @hide 171 */ ControlAction(@onNull String templateId, @Nullable String challengeValue)172 ControlAction(@NonNull String templateId, @Nullable String challengeValue) { 173 Preconditions.checkNotNull(templateId); 174 mTemplateId = templateId; 175 mChallengeValue = challengeValue; 176 } 177 178 /** 179 * @hide 180 */ ControlAction(Bundle b)181 ControlAction(Bundle b) { 182 mTemplateId = b.getString(KEY_TEMPLATE_ID); 183 mChallengeValue = b.getString(KEY_CHALLENGE_VALUE); 184 } 185 186 /** 187 * The identifier of the {@link ControlTemplate} that originated this action 188 */ 189 @NonNull getTemplateId()190 public String getTemplateId() { 191 return mTemplateId; 192 } 193 194 /** 195 * The challenge value used to authenticate certain actions, if available. 196 */ 197 @Nullable getChallengeValue()198 public String getChallengeValue() { 199 return mChallengeValue; 200 } 201 202 /** 203 * Obtain a {@link Bundle} describing this object populated with data. 204 * 205 * Implementations in subclasses should populate the {@link Bundle} returned by 206 * {@link ControlAction}. 207 * @return a {@link Bundle} containing the data that represents this object. 208 * @hide 209 */ 210 @CallSuper 211 @NonNull getDataBundle()212 Bundle getDataBundle() { 213 Bundle b = new Bundle(); 214 b.putInt(KEY_ACTION_TYPE, getActionType()); 215 b.putString(KEY_TEMPLATE_ID, mTemplateId); 216 b.putString(KEY_CHALLENGE_VALUE, mChallengeValue); 217 return b; 218 } 219 220 /** 221 * @param bundle 222 * @return 223 * @hide 224 */ 225 @NonNull createActionFromBundle(@onNull Bundle bundle)226 static ControlAction createActionFromBundle(@NonNull Bundle bundle) { 227 if (bundle == null) { 228 Log.e(TAG, "Null bundle"); 229 return ERROR_ACTION; 230 } 231 int type = bundle.getInt(KEY_ACTION_TYPE, TYPE_ERROR); 232 try { 233 switch (type) { 234 case TYPE_BOOLEAN: 235 return new BooleanAction(bundle); 236 case TYPE_FLOAT: 237 return new FloatAction(bundle); 238 case TYPE_MODE: 239 return new ModeAction(bundle); 240 case TYPE_COMMAND: 241 return new CommandAction(bundle); 242 case TYPE_ERROR: 243 default: 244 return ERROR_ACTION; 245 } 246 } catch (Exception e) { 247 Log.e(TAG, "Error creating action", e); 248 return ERROR_ACTION; 249 } 250 } 251 252 /** 253 * Returns a singleton {@link ControlAction} used for indicating an error in unparceling. 254 */ 255 @NonNull getErrorAction()256 public static ControlAction getErrorAction() { 257 return ERROR_ACTION; 258 } 259 } 260