/* * Copyright (C) 2024 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.appsindexer; import android.annotation.NonNull; import android.app.appsearch.PackageIdentifier; import android.app.appsearch.exceptions.AppSearchException; import android.content.Context; import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.util.ArrayMap; import android.util.ArraySet; import android.util.Log; import com.android.internal.annotations.VisibleForTesting; import java.io.Closeable; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.Objects; import java.util.Set; /** * Interactions with PackageManager and AppSearch. * *

This class is NOT thread-safe. * * @hide */ public final class AppsIndexerImpl implements Closeable { static final String TAG = "AppSearchAppsIndexerImpl"; private final Context mContext; private final AppSearchHelper mAppSearchHelper; public AppsIndexerImpl(@NonNull Context context) throws AppSearchException { mContext = Objects.requireNonNull(context); mAppSearchHelper = AppSearchHelper.createAppSearchHelper(context); } /** * Checks PackageManager and AppSearch to sync the Apps Index in AppSearch. * *

It deletes removed apps, inserts newly-added ones, and updates existing ones in the App * corpus in AppSearch. * * @param settings contains update timestamps that help the indexer determine which apps were * updated. */ @VisibleForTesting public void doUpdate(@NonNull AppsIndexerSettings settings) throws AppSearchException { Objects.requireNonNull(settings); long currentTimeMillis = System.currentTimeMillis(); PackageManager packageManager = mContext.getPackageManager(); // Search AppSearch for MobileApplication objects to get a "current" list of indexed apps. Map appUpdatedTimestamps = mAppSearchHelper.getAppsFromAppSearch(); Map launchablePackages = AppsUtil.getLaunchablePackages(packageManager); Set packageInfos = launchablePackages.keySet(); Map packagesToBeAddedOrUpdated = new ArrayMap<>(); long mostRecentAppUpdatedTimestampMillis = settings.getLastAppUpdateTimestampMillis(); // Prepare a set of current app IDs for efficient lookup Set currentAppIds = new ArraySet<>(); for (PackageInfo packageInfo : packageInfos) { currentAppIds.add(packageInfo.packageName); // Update the most recent timestamp as we iterate if (packageInfo.lastUpdateTime > mostRecentAppUpdatedTimestampMillis) { mostRecentAppUpdatedTimestampMillis = packageInfo.lastUpdateTime; } Long storedUpdateTime = appUpdatedTimestamps.get(packageInfo.packageName); if (storedUpdateTime == null || packageInfo.lastUpdateTime != storedUpdateTime) { // Added or updated packagesToBeAddedOrUpdated.put(packageInfo, launchablePackages.get(packageInfo)); } } try { if (!currentAppIds.equals(appUpdatedTimestamps.keySet())) { // The current list of apps in AppSearch does not match what is in PackageManager. // This means this is the first sync, an app was removed, or an app was added. In // all cases, we need to call setSchema to keep AppSearch in sync with // PackageManager. List packageIdentifiers = new ArrayList<>(); for (PackageInfo packageInfo : packageInfos) { // We get certificates here as getting the certificates during the previous for // loop would be wasteful if we end up not needing to call set schema byte[] certificate = AppsUtil.getCertificate(packageInfo); if (certificate == null) { Log.e(TAG, "Certificate not found for package: " + packageInfo.packageName); continue; } packageIdentifiers.add( new PackageIdentifier(packageInfo.packageName, certificate)); } // The certificate is necessary along with the package name as it is used in // visibility settings. mAppSearchHelper.setSchemasForPackages(packageIdentifiers); } if (!packagesToBeAddedOrUpdated.isEmpty()) { mAppSearchHelper.indexApps( AppsUtil.buildAppsFromPackageInfos( packageManager, packagesToBeAddedOrUpdated)); } settings.setLastAppUpdateTimestampMillis(mostRecentAppUpdatedTimestampMillis); settings.setLastUpdateTimestampMillis(currentTimeMillis); } catch (AppSearchException e) { // Reset the last update time stamp and app update timestamp so we can try again later. settings.reset(); throw e; } } /** Shuts down the {@link AppsIndexerImpl} and its {@link AppSearchHelper}. */ @Override public void close() { mAppSearchHelper.close(); } }