/*
* 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();
}
}