1 /* 2 * Copyright (C) 2020 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.view.translation; 18 19 import android.annotation.CallbackExecutor; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.annotation.SystemService; 23 import android.annotation.WorkerThread; 24 import android.app.PendingIntent; 25 import android.content.Context; 26 import android.content.pm.ParceledListSlice; 27 import android.os.Binder; 28 import android.os.Bundle; 29 import android.os.Handler; 30 import android.os.IRemoteCallback; 31 import android.os.Looper; 32 import android.os.RemoteException; 33 import android.os.SynchronousResultReceiver; 34 import android.util.ArrayMap; 35 import android.util.ArraySet; 36 import android.util.IntArray; 37 import android.util.Log; 38 import android.util.Pair; 39 40 import com.android.internal.annotations.GuardedBy; 41 import com.android.internal.util.SyncResultReceiver; 42 43 import java.security.SecureRandom; 44 import java.util.ArrayList; 45 import java.util.Collections; 46 import java.util.Map; 47 import java.util.Objects; 48 import java.util.Set; 49 import java.util.concurrent.Executor; 50 import java.util.concurrent.TimeoutException; 51 import java.util.concurrent.atomic.AtomicInteger; 52 import java.util.function.Consumer; 53 54 /** 55 * The {@link TranslationManager} class provides ways for apps to integrate and use the 56 * translation framework. 57 * 58 * <p>The TranslationManager manages {@link Translator}s and help bridge client calls to 59 * the server translation service </p> 60 */ 61 @SystemService(Context.TRANSLATION_MANAGER_SERVICE) 62 public final class TranslationManager { 63 64 private static final String TAG = "TranslationManager"; 65 66 /** 67 * Timeout for calls to system_server, default 1 minute. 68 */ 69 static final int SYNC_CALLS_TIMEOUT_MS = 60_000; 70 /** 71 * The result code from result receiver success. 72 * @hide 73 */ 74 public static final int STATUS_SYNC_CALL_SUCCESS = 1; 75 /** 76 * The result code from result receiver fail. 77 * @hide 78 */ 79 public static final int STATUS_SYNC_CALL_FAIL = 2; 80 81 /** 82 * Name of the extra used to pass the translation capabilities. 83 * @hide 84 */ 85 public static final String EXTRA_CAPABILITIES = "translation_capabilities"; 86 87 @GuardedBy("mLock") 88 private final ArrayMap<Pair<Integer, Integer>, ArrayList<PendingIntent>> 89 mTranslationCapabilityUpdateListeners = new ArrayMap<>(); 90 91 @GuardedBy("mLock") 92 private final Map<Consumer<TranslationCapability>, IRemoteCallback> mCapabilityCallbacks = 93 new ArrayMap<>(); 94 95 // TODO(b/158778794): make the session ids truly globally unique across processes 96 private static final SecureRandom ID_GENERATOR = new SecureRandom(); 97 private final Object mLock = new Object(); 98 99 @NonNull 100 private final Context mContext; 101 102 private final ITranslationManager mService; 103 104 @NonNull 105 @GuardedBy("mLock") 106 private final IntArray mTranslatorIds = new IntArray(); 107 108 @NonNull 109 private final Handler mHandler; 110 111 private static final AtomicInteger sAvailableRequestId = new AtomicInteger(1); 112 113 /** 114 * @hide 115 */ TranslationManager(@onNull Context context, ITranslationManager service)116 public TranslationManager(@NonNull Context context, ITranslationManager service) { 117 mContext = Objects.requireNonNull(context, "context cannot be null"); 118 mService = service; 119 120 mHandler = Handler.createAsync(Looper.getMainLooper()); 121 } 122 123 /** 124 * Creates an on-device Translator for natural language translation. 125 * 126 * <p>In Android 12, this method provided the same cached Translator object when given the 127 * same TranslationContext object. Do not use a Translator destroyed elsewhere as this will 128 * cause an exception on Android 12. 129 * 130 * <p>In later versions, this method never returns a cached Translator. 131 * 132 * @param translationContext {@link TranslationContext} containing the specs for creating the 133 * Translator. 134 * @param executor Executor to run callback operations 135 * @param callback {@link Consumer} to receive the translator. A {@code null} value is returned 136 * if the service could not create the translator. 137 */ createOnDeviceTranslator(@onNull TranslationContext translationContext, @NonNull @CallbackExecutor Executor executor, @NonNull Consumer<Translator> callback)138 public void createOnDeviceTranslator(@NonNull TranslationContext translationContext, 139 @NonNull @CallbackExecutor Executor executor, @NonNull Consumer<Translator> callback) { 140 Objects.requireNonNull(translationContext, "translationContext cannot be null"); 141 Objects.requireNonNull(executor, "executor cannot be null"); 142 Objects.requireNonNull(callback, "callback cannot be null"); 143 144 synchronized (mLock) { 145 int translatorId; 146 do { 147 translatorId = Math.abs(ID_GENERATOR.nextInt()); 148 } while (translatorId == 0 || mTranslatorIds.indexOf(translatorId) >= 0); 149 final int tId = translatorId; 150 151 new Translator(mContext, translationContext, tId, this, mHandler, mService, 152 translator -> { 153 if (translator == null) { 154 Binder.withCleanCallingIdentity( 155 () -> executor.execute(() -> callback.accept(null))); 156 return; 157 } 158 159 synchronized (mLock) { 160 mTranslatorIds.add(tId); 161 } 162 Binder.withCleanCallingIdentity( 163 () -> executor.execute(() -> callback.accept(translator))); 164 }); 165 } 166 } 167 168 /** 169 * Creates an on-device Translator for natural language translation. 170 * 171 * <p><strong>NOTE: </strong>Call on a worker thread. 172 * 173 * @removed use {@link #createOnDeviceTranslator(TranslationContext, Executor, Consumer)} 174 * instead. 175 * 176 * @param translationContext {@link TranslationContext} containing the specs for creating the 177 * Translator. 178 */ 179 @Deprecated 180 @Nullable 181 @WorkerThread createOnDeviceTranslator(@onNull TranslationContext translationContext)182 public Translator createOnDeviceTranslator(@NonNull TranslationContext translationContext) { 183 Objects.requireNonNull(translationContext, "translationContext cannot be null"); 184 185 synchronized (mLock) { 186 int translatorId; 187 do { 188 translatorId = Math.abs(ID_GENERATOR.nextInt()); 189 } while (translatorId == 0 || mTranslatorIds.indexOf(translatorId) >= 0); 190 191 final Translator newTranslator = new Translator(mContext, translationContext, 192 translatorId, this, mHandler, mService); 193 // Start the Translator session and wait for the result 194 newTranslator.start(); 195 try { 196 if (!newTranslator.isSessionCreated()) { 197 return null; 198 } 199 mTranslatorIds.add(translatorId); 200 return newTranslator; 201 } catch (Translator.ServiceBinderReceiver.TimeoutException e) { 202 // TODO(b/176464808): maybe make SyncResultReceiver.TimeoutException constructor 203 // public and use it. 204 Log.e(TAG, "Timed out getting create session: " + e); 205 return null; 206 } 207 } 208 } 209 210 /** @removed Use {@link #createOnDeviceTranslator(TranslationContext)} */ 211 @Deprecated 212 @Nullable 213 @WorkerThread createTranslator(@onNull TranslationContext translationContext)214 public Translator createTranslator(@NonNull TranslationContext translationContext) { 215 return createOnDeviceTranslator(translationContext); 216 } 217 218 /** 219 * Returns a set of {@link TranslationCapability}s describing the capabilities for on-device 220 * {@link Translator}s. 221 * 222 * <p>These translation capabilities contains a source and target {@link TranslationSpec} 223 * representing the data expected for both ends of translation process. The capabilities 224 * provides the information and limitations for generating a {@link TranslationContext}. 225 * The context object can then be used by 226 * {@link #createOnDeviceTranslator(TranslationContext, Executor, Consumer)} to obtain a 227 * {@link Translator} for translations.</p> 228 * 229 * <p><strong>NOTE: </strong>Call on a worker thread. 230 * 231 * @param sourceFormat data format for the input data to be translated. 232 * @param targetFormat data format for the expected translated output data. 233 * @return A set of {@link TranslationCapability}s. 234 */ 235 @NonNull 236 @WorkerThread getOnDeviceTranslationCapabilities( @ranslationSpec.DataFormat int sourceFormat, @TranslationSpec.DataFormat int targetFormat)237 public Set<TranslationCapability> getOnDeviceTranslationCapabilities( 238 @TranslationSpec.DataFormat int sourceFormat, 239 @TranslationSpec.DataFormat int targetFormat) { 240 try { 241 final SynchronousResultReceiver receiver = new SynchronousResultReceiver(); 242 mService.onTranslationCapabilitiesRequest(sourceFormat, targetFormat, receiver, 243 mContext.getUserId()); 244 final SynchronousResultReceiver.Result result = 245 receiver.awaitResult(SYNC_CALLS_TIMEOUT_MS); 246 if (result.resultCode != STATUS_SYNC_CALL_SUCCESS) { 247 return Collections.emptySet(); 248 } 249 ParceledListSlice<TranslationCapability> listSlice = 250 result.bundle.getParcelable(EXTRA_CAPABILITIES, android.content.pm.ParceledListSlice.class); 251 ArraySet<TranslationCapability> capabilities = 252 new ArraySet<>(listSlice == null ? null : listSlice.getList()); 253 return capabilities; 254 } catch (RemoteException e) { 255 throw e.rethrowFromSystemServer(); 256 } catch (TimeoutException e) { 257 Log.e(TAG, "Timed out getting supported translation capabilities: " + e); 258 return Collections.emptySet(); 259 } 260 } 261 262 /** @removed Use {@link #getOnDeviceTranslationCapabilities(int, int)} */ 263 @Deprecated 264 @NonNull 265 @WorkerThread getTranslationCapabilities( @ranslationSpec.DataFormat int sourceFormat, @TranslationSpec.DataFormat int targetFormat)266 public Set<TranslationCapability> getTranslationCapabilities( 267 @TranslationSpec.DataFormat int sourceFormat, 268 @TranslationSpec.DataFormat int targetFormat) { 269 return getOnDeviceTranslationCapabilities(sourceFormat, targetFormat); 270 } 271 272 /** 273 * Adds a {@link TranslationCapability} Consumer to listen for updates on states of on-device 274 * {@link TranslationCapability}s. 275 * 276 * @param capabilityListener a {@link TranslationCapability} Consumer to receive the updated 277 * {@link TranslationCapability} from the on-device translation service. 278 */ addOnDeviceTranslationCapabilityUpdateListener( @onNull @allbackExecutor Executor executor, @NonNull Consumer<TranslationCapability> capabilityListener)279 public void addOnDeviceTranslationCapabilityUpdateListener( 280 @NonNull @CallbackExecutor Executor executor, 281 @NonNull Consumer<TranslationCapability> capabilityListener) { 282 Objects.requireNonNull(executor, "executor should not be null"); 283 Objects.requireNonNull(capabilityListener, "capability listener should not be null"); 284 285 synchronized (mLock) { 286 if (mCapabilityCallbacks.containsKey(capabilityListener)) { 287 Log.w(TAG, "addOnDeviceTranslationCapabilityUpdateListener: the listener for " 288 + capabilityListener + " already registered; ignoring."); 289 return; 290 } 291 final IRemoteCallback remoteCallback = new TranslationCapabilityRemoteCallback(executor, 292 capabilityListener); 293 try { 294 mService.registerTranslationCapabilityCallback(remoteCallback, 295 mContext.getUserId()); 296 } catch (RemoteException e) { 297 throw e.rethrowFromSystemServer(); 298 } 299 mCapabilityCallbacks.put(capabilityListener, remoteCallback); 300 } 301 } 302 303 304 /** 305 * @removed Use {@link TranslationManager#addOnDeviceTranslationCapabilityUpdateListener( 306 * java.util.concurrent.Executor, java.util.function.Consumer)} 307 */ 308 @Deprecated addOnDeviceTranslationCapabilityUpdateListener( @ranslationSpec.DataFormat int sourceFormat, @TranslationSpec.DataFormat int targetFormat, @NonNull PendingIntent pendingIntent)309 public void addOnDeviceTranslationCapabilityUpdateListener( 310 @TranslationSpec.DataFormat int sourceFormat, 311 @TranslationSpec.DataFormat int targetFormat, 312 @NonNull PendingIntent pendingIntent) { 313 Objects.requireNonNull(pendingIntent, "pending intent should not be null"); 314 315 synchronized (mLock) { 316 final Pair<Integer, Integer> formatPair = new Pair<>(sourceFormat, targetFormat); 317 mTranslationCapabilityUpdateListeners.computeIfAbsent(formatPair, 318 (formats) -> new ArrayList<>()).add(pendingIntent); 319 } 320 } 321 322 /** 323 * @removed Use {@link TranslationManager#addOnDeviceTranslationCapabilityUpdateListener( 324 * java.util.concurrent.Executor, java.util.function.Consumer)} 325 */ 326 @Deprecated addTranslationCapabilityUpdateListener( @ranslationSpec.DataFormat int sourceFormat, @TranslationSpec.DataFormat int targetFormat, @NonNull PendingIntent pendingIntent)327 public void addTranslationCapabilityUpdateListener( 328 @TranslationSpec.DataFormat int sourceFormat, 329 @TranslationSpec.DataFormat int targetFormat, 330 @NonNull PendingIntent pendingIntent) { 331 addOnDeviceTranslationCapabilityUpdateListener(sourceFormat, targetFormat, pendingIntent); 332 } 333 334 /** 335 * Removes a {@link TranslationCapability} Consumer to listen for updates on states of 336 * on-device {@link TranslationCapability}s. 337 * 338 * @param capabilityListener the {@link TranslationCapability} Consumer to unregister 339 */ removeOnDeviceTranslationCapabilityUpdateListener( @onNull Consumer<TranslationCapability> capabilityListener)340 public void removeOnDeviceTranslationCapabilityUpdateListener( 341 @NonNull Consumer<TranslationCapability> capabilityListener) { 342 Objects.requireNonNull(capabilityListener, "capability callback should not be null"); 343 344 synchronized (mLock) { 345 final IRemoteCallback remoteCallback = mCapabilityCallbacks.get(capabilityListener); 346 if (remoteCallback == null) { 347 Log.w(TAG, "removeOnDeviceTranslationCapabilityUpdateListener: the capability " 348 + "listener not found; ignoring."); 349 return; 350 } 351 try { 352 mService.unregisterTranslationCapabilityCallback(remoteCallback, 353 mContext.getUserId()); 354 } catch (RemoteException e) { 355 throw e.rethrowFromSystemServer(); 356 } 357 mCapabilityCallbacks.remove(capabilityListener); 358 } 359 } 360 361 /** 362 * @removed Use {@link #removeOnDeviceTranslationCapabilityUpdateListener( 363 * java.util.function.Consumer)}. 364 */ 365 @Deprecated removeOnDeviceTranslationCapabilityUpdateListener( @ranslationSpec.DataFormat int sourceFormat, @TranslationSpec.DataFormat int targetFormat, @NonNull PendingIntent pendingIntent)366 public void removeOnDeviceTranslationCapabilityUpdateListener( 367 @TranslationSpec.DataFormat int sourceFormat, 368 @TranslationSpec.DataFormat int targetFormat, 369 @NonNull PendingIntent pendingIntent) { 370 Objects.requireNonNull(pendingIntent, "pending intent should not be null"); 371 372 synchronized (mLock) { 373 final Pair<Integer, Integer> formatPair = new Pair<>(sourceFormat, targetFormat); 374 if (mTranslationCapabilityUpdateListeners.containsKey(formatPair)) { 375 final ArrayList<PendingIntent> intents = 376 mTranslationCapabilityUpdateListeners.get(formatPair); 377 if (intents.contains(pendingIntent)) { 378 intents.remove(pendingIntent); 379 } else { 380 Log.w(TAG, "pending intent=" + pendingIntent + " does not exist in " 381 + "mTranslationCapabilityUpdateListeners"); 382 } 383 } else { 384 Log.w(TAG, "format pair=" + formatPair + " does not exist in " 385 + "mTranslationCapabilityUpdateListeners"); 386 } 387 } 388 } 389 390 /** 391 * @removed Use {@link #removeOnDeviceTranslationCapabilityUpdateListener( 392 * java.util.function.Consumer)}. 393 */ 394 @Deprecated removeTranslationCapabilityUpdateListener( @ranslationSpec.DataFormat int sourceFormat, @TranslationSpec.DataFormat int targetFormat, @NonNull PendingIntent pendingIntent)395 public void removeTranslationCapabilityUpdateListener( 396 @TranslationSpec.DataFormat int sourceFormat, 397 @TranslationSpec.DataFormat int targetFormat, 398 @NonNull PendingIntent pendingIntent) { 399 removeOnDeviceTranslationCapabilityUpdateListener( 400 sourceFormat, targetFormat, pendingIntent); 401 } 402 403 /** 404 * Returns an immutable PendingIntent which can be used to launch an activity to view/edit 405 * on-device translation settings. 406 * 407 * @return An immutable PendingIntent or {@code null} if one of reason met: 408 * <ul> 409 * <li>Device manufacturer (OEM) does not provide TranslationService.</li> 410 * <li>The TranslationService doesn't provide the Settings.</li> 411 * </ul> 412 **/ 413 @Nullable getOnDeviceTranslationSettingsActivityIntent()414 public PendingIntent getOnDeviceTranslationSettingsActivityIntent() { 415 final SyncResultReceiver resultReceiver = new SyncResultReceiver(SYNC_CALLS_TIMEOUT_MS); 416 try { 417 mService.getServiceSettingsActivity(resultReceiver, mContext.getUserId()); 418 } catch (RemoteException e) { 419 throw e.rethrowFromSystemServer(); 420 } 421 try { 422 return resultReceiver.getParcelableResult(); 423 } catch (SyncResultReceiver.TimeoutException e) { 424 Log.e(TAG, "Fail to get translation service settings activity."); 425 return null; 426 } 427 } 428 429 /** @removed Use {@link #getOnDeviceTranslationSettingsActivityIntent()} */ 430 @Deprecated 431 @Nullable getTranslationSettingsActivityIntent()432 public PendingIntent getTranslationSettingsActivityIntent() { 433 return getOnDeviceTranslationSettingsActivityIntent(); 434 } 435 removeTranslator(int id)436 void removeTranslator(int id) { 437 synchronized (mLock) { 438 int index = mTranslatorIds.indexOf(id); 439 if (index >= 0) { 440 mTranslatorIds.remove(index); 441 } 442 } 443 } 444 getAvailableRequestId()445 AtomicInteger getAvailableRequestId() { 446 synchronized (mLock) { 447 return sAvailableRequestId; 448 } 449 } 450 451 private static class TranslationCapabilityRemoteCallback extends 452 IRemoteCallback.Stub { 453 private final Executor mExecutor; 454 private final Consumer<TranslationCapability> mListener; 455 TranslationCapabilityRemoteCallback(Executor executor, Consumer<TranslationCapability> listener)456 TranslationCapabilityRemoteCallback(Executor executor, 457 Consumer<TranslationCapability> listener) { 458 mExecutor = executor; 459 mListener = listener; 460 } 461 462 @Override sendResult(Bundle bundle)463 public void sendResult(Bundle bundle) { 464 Binder.withCleanCallingIdentity( 465 () -> mExecutor.execute(() -> onTranslationCapabilityUpdate(bundle))); 466 } 467 onTranslationCapabilityUpdate(Bundle bundle)468 private void onTranslationCapabilityUpdate(Bundle bundle) { 469 TranslationCapability capability = 470 (TranslationCapability) bundle.getParcelable(EXTRA_CAPABILITIES, android.view.translation.TranslationCapability.class); 471 mListener.accept(capability); 472 } 473 } 474 } 475