1 /* 2 * Copyright (C) 2014 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.test.voiceenrollment; 18 19 import android.annotation.Nullable; 20 import android.content.Context; 21 import android.hardware.soundtrigger.KeyphraseEnrollmentInfo; 22 import android.hardware.soundtrigger.SoundTrigger; 23 import android.hardware.soundtrigger.SoundTrigger.Keyphrase; 24 import android.hardware.soundtrigger.SoundTrigger.KeyphraseSoundModel; 25 import android.os.RemoteException; 26 import android.os.ServiceManager; 27 import android.service.voice.AlwaysOnHotwordDetector; 28 import android.util.Log; 29 30 import com.android.internal.app.IVoiceInteractionManagerService; 31 32 /** 33 * Utility class for the enrollment operations like enroll;re-enroll & un-enroll. 34 */ 35 public class EnrollmentUtil { 36 private static final String TAG = "TestEnrollmentUtil"; 37 38 /** 39 * Activity Action: Show activity for managing the keyphrases for hotword detection. 40 * This needs to be defined by an activity that supports enrolling users for hotword/keyphrase 41 * detection. 42 */ 43 public static final String ACTION_MANAGE_VOICE_KEYPHRASES = 44 KeyphraseEnrollmentInfo.ACTION_MANAGE_VOICE_KEYPHRASES; 45 46 /** 47 * Intent extra: The intent extra for the specific manage action that needs to be performed. 48 * Possible values are {@link AlwaysOnHotwordDetector#MANAGE_ACTION_ENROLL}, 49 * {@link AlwaysOnHotwordDetector#MANAGE_ACTION_RE_ENROLL} 50 * or {@link AlwaysOnHotwordDetector#MANAGE_ACTION_UN_ENROLL}. 51 */ 52 public static final String EXTRA_VOICE_KEYPHRASE_ACTION = 53 KeyphraseEnrollmentInfo.EXTRA_VOICE_KEYPHRASE_ACTION; 54 55 /** 56 * Intent extra: The hint text to be shown on the voice keyphrase management UI. 57 */ 58 public static final String EXTRA_VOICE_KEYPHRASE_HINT_TEXT = 59 KeyphraseEnrollmentInfo.EXTRA_VOICE_KEYPHRASE_HINT_TEXT; 60 /** 61 * Intent extra: The voice locale to use while managing the keyphrase. 62 */ 63 public static final String EXTRA_VOICE_KEYPHRASE_LOCALE = 64 KeyphraseEnrollmentInfo.EXTRA_VOICE_KEYPHRASE_LOCALE; 65 66 /** Simple recognition of the key phrase */ 67 public static final int RECOGNITION_MODE_VOICE_TRIGGER = 68 SoundTrigger.RECOGNITION_MODE_VOICE_TRIGGER; 69 /** Trigger only if one user is identified */ 70 public static final int RECOGNITION_MODE_USER_IDENTIFICATION = 71 SoundTrigger.RECOGNITION_MODE_USER_IDENTIFICATION; 72 73 private final IVoiceInteractionManagerService mModelManagementService; 74 75 public EnrollmentUtil() { 76 mModelManagementService = IVoiceInteractionManagerService.Stub.asInterface( 77 ServiceManager.getService(Context.VOICE_INTERACTION_MANAGER_SERVICE)); 78 } 79 80 /** 81 * Adds/Updates a sound model. 82 * The sound model must contain a valid UUID, 83 * exactly 1 keyphrase, 84 * and users for which the keyphrase is valid - typically the current user. 85 * 86 * @param soundModel The sound model to add/update. 87 * @return {@code true} if the call succeeds, {@code false} otherwise. 88 */ 89 public boolean addOrUpdateSoundModel(KeyphraseSoundModel soundModel) { 90 if (!verifyKeyphraseSoundModel(soundModel)) { 91 return false; 92 } 93 94 int status = SoundTrigger.STATUS_ERROR; 95 try { 96 status = mModelManagementService.updateKeyphraseSoundModel(soundModel); 97 } catch (RemoteException e) { 98 Log.e(TAG, "RemoteException in updateKeyphraseSoundModel", e); 99 } 100 return status == SoundTrigger.STATUS_OK; 101 } 102 103 /** 104 * Gets the sound model for the given keyphrase, null if none exists. 105 * This should be used for re-enrollment purposes. 106 * If a sound model for a given keyphrase exists, and it needs to be updated, 107 * it should be obtained using this method, updated and then passed in to 108 * {@link #addOrUpdateSoundModel(KeyphraseSoundModel)} without changing the IDs. 109 * 110 * @param keyphraseId The keyphrase ID to look-up the sound model for. 111 * @param bcp47Locale The locale for with to look up the sound model for. 112 * @return The sound model if one was found, null otherwise. 113 */ 114 @Nullable 115 public KeyphraseSoundModel getSoundModel(int keyphraseId, String bcp47Locale) { 116 if (keyphraseId <= 0) { 117 Log.e(TAG, "Keyphrase must have a valid ID"); 118 return null; 119 } 120 121 KeyphraseSoundModel model = null; 122 try { 123 model = mModelManagementService.getKeyphraseSoundModel(keyphraseId, bcp47Locale); 124 } catch (RemoteException e) { 125 Log.e(TAG, "RemoteException in updateKeyphraseSoundModel"); 126 } 127 128 if (model == null) { 129 Log.w(TAG, "No models present for the gien keyphrase ID"); 130 return null; 131 } else { 132 return model; 133 } 134 } 135 136 /** 137 * Deletes the sound model for the given keyphrase id. 138 * 139 * @param keyphraseId The keyphrase ID to look-up the sound model for. 140 * @return {@code true} if the call succeeds, {@code false} otherwise. 141 */ 142 public boolean deleteSoundModel(int keyphraseId, String bcp47Locale) { 143 if (keyphraseId <= 0) { 144 Log.e(TAG, "Keyphrase must have a valid ID"); 145 return false; 146 } 147 148 int status = SoundTrigger.STATUS_ERROR; 149 try { 150 status = mModelManagementService.deleteKeyphraseSoundModel(keyphraseId, bcp47Locale); 151 } catch (RemoteException e) { 152 Log.e(TAG, "RemoteException in updateKeyphraseSoundModel"); 153 } 154 return status == SoundTrigger.STATUS_OK; 155 } 156 157 private boolean verifyKeyphraseSoundModel(KeyphraseSoundModel soundModel) { 158 if (soundModel == null) { 159 Log.e(TAG, "KeyphraseSoundModel must be non-null"); 160 return false; 161 } 162 if (soundModel.uuid == null) { 163 Log.e(TAG, "KeyphraseSoundModel must have a UUID"); 164 return false; 165 } 166 if (soundModel.data == null) { 167 Log.e(TAG, "KeyphraseSoundModel must have data"); 168 return false; 169 } 170 if (soundModel.keyphrases == null || soundModel.keyphrases.length != 1) { 171 Log.e(TAG, "Keyphrase must be exactly 1"); 172 return false; 173 } 174 Keyphrase keyphrase = soundModel.keyphrases[0]; 175 if (keyphrase.id <= 0) { 176 Log.e(TAG, "Keyphrase must have a valid ID"); 177 return false; 178 } 179 if (keyphrase.recognitionModes < 0) { 180 Log.e(TAG, "Recognition modes must be valid"); 181 return false; 182 } 183 if (keyphrase.locale == null) { 184 Log.e(TAG, "Locale must not be null"); 185 return false; 186 } 187 if (keyphrase.text == null) { 188 Log.e(TAG, "Text must not be null"); 189 return false; 190 } 191 if (keyphrase.users == null || keyphrase.users.length == 0) { 192 Log.e(TAG, "Keyphrase must have valid user(s)"); 193 return false; 194 } 195 return true; 196 } 197 } 198