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 EnrollmentUtil()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 */ addOrUpdateSoundModel(KeyphraseSoundModel soundModel)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 getSoundModel(int keyphraseId, String bcp47Locale)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 @Nullable deleteSoundModel(int keyphraseId, String bcp47Locale)143 public boolean deleteSoundModel(int keyphraseId, String bcp47Locale) { 144 if (keyphraseId <= 0) { 145 Log.e(TAG, "Keyphrase must have a valid ID"); 146 return false; 147 } 148 149 int status = SoundTrigger.STATUS_ERROR; 150 try { 151 status = mModelManagementService.deleteKeyphraseSoundModel(keyphraseId, bcp47Locale); 152 } catch (RemoteException e) { 153 Log.e(TAG, "RemoteException in updateKeyphraseSoundModel"); 154 } 155 return status == SoundTrigger.STATUS_OK; 156 } 157 verifyKeyphraseSoundModel(KeyphraseSoundModel soundModel)158 private boolean verifyKeyphraseSoundModel(KeyphraseSoundModel soundModel) { 159 if (soundModel == null) { 160 Log.e(TAG, "KeyphraseSoundModel must be non-null"); 161 return false; 162 } 163 if (soundModel.uuid == null) { 164 Log.e(TAG, "KeyphraseSoundModel must have a UUID"); 165 return false; 166 } 167 if (soundModel.data == null) { 168 Log.e(TAG, "KeyphraseSoundModel must have data"); 169 return false; 170 } 171 if (soundModel.keyphrases == null || soundModel.keyphrases.length != 1) { 172 Log.e(TAG, "Keyphrase must be exactly 1"); 173 return false; 174 } 175 Keyphrase keyphrase = soundModel.keyphrases[0]; 176 if (keyphrase.id <= 0) { 177 Log.e(TAG, "Keyphrase must have a valid ID"); 178 return false; 179 } 180 if (keyphrase.recognitionModes < 0) { 181 Log.e(TAG, "Recognition modes must be valid"); 182 return false; 183 } 184 if (keyphrase.locale == null) { 185 Log.e(TAG, "Locale must not be null"); 186 return false; 187 } 188 if (keyphrase.text == null) { 189 Log.e(TAG, "Text must not be null"); 190 return false; 191 } 192 if (keyphrase.users == null || keyphrase.users.length == 0) { 193 Log.e(TAG, "Keyphrase must have valid user(s)"); 194 return false; 195 } 196 return true; 197 } 198 } 199