| /* |
| * Copyright (C) 2020 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package com.android.server.appsearch; |
| |
| import android.annotation.NonNull; |
| import android.app.appsearch.AppSearchBatchResult; |
| import android.app.appsearch.AppSearchResult; |
| import android.app.appsearch.AppSearchSchema; |
| import android.app.appsearch.GenericDocument; |
| import android.app.appsearch.IAppSearchManager; |
| import android.app.appsearch.SearchResultPage; |
| import android.app.appsearch.SearchSpec; |
| import android.app.appsearch.exceptions.AppSearchException; |
| import android.content.Context; |
| import android.os.Binder; |
| import android.os.Bundle; |
| import android.os.UserHandle; |
| import android.util.ArraySet; |
| |
| import com.android.internal.infra.AndroidFuture; |
| import com.android.internal.util.Preconditions; |
| import com.android.server.SystemService; |
| import com.android.server.appsearch.external.localstorage.AppSearchImpl; |
| |
| import java.io.IOException; |
| import java.util.List; |
| import java.util.Set; |
| |
| /** |
| * TODO(b/142567528): add comments when implement this class |
| */ |
| public class AppSearchManagerService extends SystemService { |
| private static final String TAG = "AppSearchManagerService"; |
| private static final char CALLING_NAME_DATABASE_DELIMITER = '$'; |
| |
| public AppSearchManagerService(Context context) { |
| super(context); |
| } |
| |
| @Override |
| public void onStart() { |
| publishBinderService(Context.APP_SEARCH_SERVICE, new Stub()); |
| } |
| |
| private class Stub extends IAppSearchManager.Stub { |
| @Override |
| public void setSchema( |
| @NonNull String databaseName, |
| @NonNull List<Bundle> schemaBundles, |
| boolean forceOverride, |
| @NonNull AndroidFuture<AppSearchResult> callback) { |
| Preconditions.checkNotNull(databaseName); |
| Preconditions.checkNotNull(schemaBundles); |
| Preconditions.checkNotNull(callback); |
| int callingUid = Binder.getCallingUidOrThrow(); |
| int callingUserId = UserHandle.getUserId(callingUid); |
| final long callingIdentity = Binder.clearCallingIdentity(); |
| try { |
| Set<AppSearchSchema> schemas = new ArraySet<>(schemaBundles.size()); |
| for (int i = 0; i < schemaBundles.size(); i++) { |
| schemas.add(new AppSearchSchema(schemaBundles.get(i))); |
| } |
| AppSearchImpl impl = ImplInstanceManager.getInstance(getContext(), callingUserId); |
| databaseName = rewriteDatabaseNameWithUid(databaseName, callingUid); |
| impl.setSchema(databaseName, schemas, forceOverride); |
| callback.complete(AppSearchResult.newSuccessfulResult(/*result=*/ null)); |
| } catch (Throwable t) { |
| callback.complete(throwableToFailedResult(t)); |
| } finally { |
| Binder.restoreCallingIdentity(callingIdentity); |
| } |
| } |
| |
| @Override |
| public void putDocuments( |
| @NonNull String databaseName, |
| @NonNull List<Bundle> documentBundles, |
| @NonNull AndroidFuture<AppSearchBatchResult> callback) { |
| Preconditions.checkNotNull(databaseName); |
| Preconditions.checkNotNull(documentBundles); |
| Preconditions.checkNotNull(callback); |
| int callingUid = Binder.getCallingUidOrThrow(); |
| int callingUserId = UserHandle.getUserId(callingUid); |
| final long callingIdentity = Binder.clearCallingIdentity(); |
| try { |
| AppSearchImpl impl = ImplInstanceManager.getInstance(getContext(), callingUserId); |
| databaseName = rewriteDatabaseNameWithUid(databaseName, callingUid); |
| AppSearchBatchResult.Builder<String, Void> resultBuilder = |
| new AppSearchBatchResult.Builder<>(); |
| for (int i = 0; i < documentBundles.size(); i++) { |
| GenericDocument document = new GenericDocument(documentBundles.get(i)); |
| try { |
| impl.putDocument(databaseName, document); |
| resultBuilder.setSuccess(document.getUri(), /*result=*/ null); |
| } catch (Throwable t) { |
| resultBuilder.setResult(document.getUri(), throwableToFailedResult(t)); |
| } |
| } |
| callback.complete(resultBuilder.build()); |
| } catch (Throwable t) { |
| callback.completeExceptionally(t); |
| } finally { |
| Binder.restoreCallingIdentity(callingIdentity); |
| } |
| } |
| |
| @Override |
| public void getDocuments(@NonNull String databaseName, @NonNull String namespace, |
| @NonNull List<String> uris, @NonNull AndroidFuture<AppSearchBatchResult> callback) { |
| Preconditions.checkNotNull(databaseName); |
| Preconditions.checkNotNull(namespace); |
| Preconditions.checkNotNull(uris); |
| Preconditions.checkNotNull(callback); |
| int callingUid = Binder.getCallingUidOrThrow(); |
| int callingUserId = UserHandle.getUserId(callingUid); |
| final long callingIdentity = Binder.clearCallingIdentity(); |
| try { |
| AppSearchImpl impl = ImplInstanceManager.getInstance(getContext(), callingUserId); |
| databaseName = rewriteDatabaseNameWithUid(databaseName, callingUid); |
| AppSearchBatchResult.Builder<String, Bundle> resultBuilder = |
| new AppSearchBatchResult.Builder<>(); |
| for (int i = 0; i < uris.size(); i++) { |
| String uri = uris.get(i); |
| try { |
| GenericDocument document = impl.getDocument(databaseName, namespace, uri); |
| resultBuilder.setSuccess(uri, document.getBundle()); |
| } catch (Throwable t) { |
| resultBuilder.setResult(uri, throwableToFailedResult(t)); |
| } |
| } |
| callback.complete(resultBuilder.build()); |
| } catch (Throwable t) { |
| callback.completeExceptionally(t); |
| } finally { |
| Binder.restoreCallingIdentity(callingIdentity); |
| } |
| } |
| |
| // TODO(sidchhabra): Do this in a threadpool. |
| // TODO(b/162450968) handle pagination after getNextPage and SearchResults is ready. |
| @Override |
| public void query( |
| @NonNull String databaseName, |
| @NonNull String queryExpression, |
| @NonNull Bundle searchSpecBundle, |
| @NonNull AndroidFuture<AppSearchResult> callback) { |
| Preconditions.checkNotNull(databaseName); |
| Preconditions.checkNotNull(queryExpression); |
| Preconditions.checkNotNull(searchSpecBundle); |
| Preconditions.checkNotNull(callback); |
| int callingUid = Binder.getCallingUidOrThrow(); |
| int callingUserId = UserHandle.getUserId(callingUid); |
| final long callingIdentity = Binder.clearCallingIdentity(); |
| try { |
| AppSearchImpl impl = ImplInstanceManager.getInstance(getContext(), callingUserId); |
| databaseName = rewriteDatabaseNameWithUid(databaseName, callingUid); |
| SearchResultPage searchResultPage = impl.query( |
| databaseName, |
| queryExpression, |
| new SearchSpec(searchSpecBundle)); |
| callback.complete( |
| AppSearchResult.newSuccessfulResult(searchResultPage.getBundle())); |
| } catch (Throwable t) { |
| callback.complete(throwableToFailedResult(t)); |
| } finally { |
| Binder.restoreCallingIdentity(callingIdentity); |
| } |
| } |
| |
| @Override |
| public void removeByUri(@NonNull String databaseName, @NonNull String namespace, |
| List<String> uris, AndroidFuture<AppSearchBatchResult> callback) { |
| Preconditions.checkNotNull(databaseName); |
| Preconditions.checkNotNull(namespace); |
| Preconditions.checkNotNull(uris); |
| Preconditions.checkNotNull(callback); |
| int callingUid = Binder.getCallingUidOrThrow(); |
| int callingUserId = UserHandle.getUserId(callingUid); |
| final long callingIdentity = Binder.clearCallingIdentity(); |
| try { |
| AppSearchImpl impl = ImplInstanceManager.getInstance(getContext(), callingUserId); |
| databaseName = rewriteDatabaseNameWithUid(databaseName, callingUid); |
| AppSearchBatchResult.Builder<String, Void> resultBuilder = |
| new AppSearchBatchResult.Builder<>(); |
| for (int i = 0; i < uris.size(); i++) { |
| String uri = uris.get(i); |
| try { |
| impl.remove(databaseName, namespace, uri); |
| resultBuilder.setSuccess(uri, /*result= */null); |
| } catch (Throwable t) { |
| resultBuilder.setResult(uri, throwableToFailedResult(t)); |
| } |
| } |
| callback.complete(resultBuilder.build()); |
| } catch (Throwable t) { |
| callback.completeExceptionally(t); |
| } finally { |
| Binder.restoreCallingIdentity(callingIdentity); |
| } |
| } |
| |
| @Override |
| public void removeByQuery( |
| @NonNull String databaseName, |
| @NonNull String queryExpression, |
| @NonNull Bundle searchSpecBundle, |
| @NonNull AndroidFuture<AppSearchResult> callback) { |
| Preconditions.checkNotNull(databaseName); |
| Preconditions.checkNotNull(queryExpression); |
| Preconditions.checkNotNull(searchSpecBundle); |
| Preconditions.checkNotNull(callback); |
| int callingUid = Binder.getCallingUidOrThrow(); |
| int callingUserId = UserHandle.getUserId(callingUid); |
| final long callingIdentity = Binder.clearCallingIdentity(); |
| try { |
| AppSearchImpl impl = ImplInstanceManager.getInstance(getContext(), callingUserId); |
| databaseName = rewriteDatabaseNameWithUid(databaseName, callingUid); |
| impl.removeByQuery(databaseName, queryExpression, new SearchSpec(searchSpecBundle)); |
| callback.complete(AppSearchResult.newSuccessfulResult(/*result= */null)); |
| } catch (Throwable t) { |
| callback.complete(throwableToFailedResult(t)); |
| } finally { |
| Binder.restoreCallingIdentity(callingIdentity); |
| } |
| } |
| |
| /** |
| * Rewrites the database name by adding a prefix of unique name for the given uid. |
| * |
| * <p>The current implementation returns the package name of the app with this uid in a |
| * format like {@code com.example.package} or {@code com.example.sharedname:5678}. |
| */ |
| @NonNull |
| private String rewriteDatabaseNameWithUid(String databaseName, int callingUid) { |
| // For regular apps, this call will return the package name. If callingUid is an |
| // android:sharedUserId, this value may be another type of name and have a :uid suffix. |
| String callingUidName = getContext().getPackageManager().getNameForUid(callingUid); |
| if (callingUidName == null) { |
| // Not sure how this is possible --- maybe app was uninstalled? |
| throw new IllegalStateException( |
| "Failed to look up package name for uid " + callingUid); |
| } |
| return callingUidName + CALLING_NAME_DATABASE_DELIMITER + databaseName; |
| } |
| |
| private <ValueType> AppSearchResult<ValueType> throwableToFailedResult( |
| @NonNull Throwable t) { |
| if (t instanceof AppSearchException) { |
| return ((AppSearchException) t).toAppSearchResult(); |
| } |
| |
| @AppSearchResult.ResultCode int resultCode; |
| if (t instanceof IllegalStateException) { |
| resultCode = AppSearchResult.RESULT_INTERNAL_ERROR; |
| } else if (t instanceof IllegalArgumentException) { |
| resultCode = AppSearchResult.RESULT_INVALID_ARGUMENT; |
| } else if (t instanceof IOException) { |
| resultCode = AppSearchResult.RESULT_IO_ERROR; |
| } else { |
| resultCode = AppSearchResult.RESULT_UNKNOWN_ERROR; |
| } |
| return AppSearchResult.newFailedResult(resultCode, t.getMessage()); |
| } |
| } |
| } |