blob: 4c847dbf9224158a7ad7bf4d6229ddc418934741 [file] [log] [blame]
/*
* Copyright (C) 2017 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 android.app;
import android.annotation.MainThread;
import android.annotation.NonNull;
import android.annotation.Nullable;
import android.annotation.SystemApi;
import android.content.Context;
import android.content.Intent;
import android.content.pm.InstantAppRequestInfo;
import android.content.pm.InstantAppResolveInfo;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.IBinder;
import android.os.IRemoteCallback;
import android.os.Looper;
import android.os.Message;
import android.os.RemoteException;
import android.os.UserHandle;
import android.util.Log;
import android.util.Slog;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.os.SomeArgs;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
/**
* Base class for implementing the resolver service.
* @hide
*/
@SystemApi
public abstract class InstantAppResolverService extends Service {
private static final boolean DEBUG_INSTANT = Build.IS_DEBUGGABLE;
private static final String TAG = "PackageManager";
/** @hide */
public static final String EXTRA_RESOLVE_INFO = "android.app.extra.RESOLVE_INFO";
/** @hide */
public static final String EXTRA_SEQUENCE = "android.app.extra.SEQUENCE";
Handler mHandler;
/**
* Called to retrieve resolve info for instant applications immediately.
*
* @param digestPrefix The hash prefix of the instant app's domain.
*
* @deprecated Should implement {@link #onGetInstantAppResolveInfo(InstantAppRequestInfo,
* InstantAppResolutionCallback)}
*/
@Deprecated
public void onGetInstantAppResolveInfo(@Nullable int[] digestPrefix, @NonNull String token,
@NonNull InstantAppResolutionCallback callback) {
throw new IllegalStateException("Must define onGetInstantAppResolveInfo");
}
/**
* Called to retrieve intent filters for instant applications from potentially expensive
* sources.
*
* @param digestPrefix The hash prefix of the instant app's domain.
*
* @deprecated Should implement {@link #onGetInstantAppIntentFilter(InstantAppRequestInfo,
* InstantAppResolutionCallback)}
*/
@Deprecated
public void onGetInstantAppIntentFilter(@Nullable int[] digestPrefix, @NonNull String token,
@NonNull InstantAppResolutionCallback callback) {
throw new IllegalStateException("Must define onGetInstantAppIntentFilter");
}
/**
* Called to retrieve resolve info for instant applications immediately. The response will be
* ignored if not provided within a reasonable time. {@link InstantAppResolveInfo}s provided
* in response to this method may be partial to request a second phase of resolution which will
* result in a subsequent call to
* {@link #onGetInstantAppIntentFilter(Intent, int[], String, InstantAppResolutionCallback)}
*
* @param sanitizedIntent The sanitized {@link Intent} used for resolution. A sanitized Intent
* is an intent with potential PII removed from the original intent.
* Fields removed include extras and the host + path of the data, if
* defined.
* @param hostDigestPrefix The hash prefix of the instant app's domain.
* @param token A unique identifier that will be provided in calls to
* {@link #onGetInstantAppIntentFilter(Intent, int[], String,
* InstantAppResolutionCallback)}
* and provided to the installer via {@link Intent#EXTRA_INSTANT_APP_TOKEN} to
* tie a single launch together.
* @param callback The {@link InstantAppResolutionCallback} to provide results to.
*
* @see InstantAppResolveInfo
*
* @deprecated Should implement {@link #onGetInstantAppResolveInfo(InstantAppRequestInfo,
* InstantAppResolutionCallback)}
*/
@Deprecated
public void onGetInstantAppResolveInfo(@NonNull Intent sanitizedIntent,
@Nullable int[] hostDigestPrefix, @NonNull String token,
@NonNull InstantAppResolutionCallback callback) {
// if not overridden, forward to old methods and filter out non-web intents
if (sanitizedIntent.isWebIntent()) {
onGetInstantAppResolveInfo(hostDigestPrefix, token, callback);
} else {
callback.onInstantAppResolveInfo(Collections.emptyList());
}
}
/**
* Called to retrieve intent filters for potentially matching instant applications. Unlike
* {@link #onGetInstantAppResolveInfo(Intent, int[], String, InstantAppResolutionCallback)},
* the response may take as long as necessary to respond. All {@link InstantAppResolveInfo}s
* provided in response to this method must be completely populated.
*
* @param sanitizedIntent The sanitized {@link Intent} used for resolution.
* @param hostDigestPrefix The hash prefix of the instant app's domain or null if no host is
* defined.
* @param token A unique identifier that was provided in
* {@link #onGetInstantAppResolveInfo(Intent, int[], String,
* InstantAppResolutionCallback)}
* and provided to the currently visible installer via
* {@link Intent#EXTRA_INSTANT_APP_TOKEN}.
* @param callback The {@link InstantAppResolutionCallback} to provide results to.
*
* @deprecated Should implement {@link #onGetInstantAppIntentFilter(InstantAppRequestInfo,
* InstantAppResolutionCallback)}
*/
@Deprecated
public void onGetInstantAppIntentFilter(@NonNull Intent sanitizedIntent,
@Nullable int[] hostDigestPrefix,
@NonNull String token, @NonNull InstantAppResolutionCallback callback) {
Log.e(TAG, "New onGetInstantAppIntentFilter is not overridden");
// if not overridden, forward to old methods and filter out non-web intents
if (sanitizedIntent.isWebIntent()) {
onGetInstantAppIntentFilter(hostDigestPrefix, token, callback);
} else {
callback.onInstantAppResolveInfo(Collections.emptyList());
}
}
/**
* Called to retrieve resolve info for instant applications immediately. The response will be
* ignored if not provided within a reasonable time. {@link InstantAppResolveInfo}s provided
* in response to this method may be partial to request a second phase of resolution which will
* result in a subsequent call to {@link #onGetInstantAppIntentFilter(Intent, int[], UserHandle,
* String, InstantAppResolutionCallback)}
*
* @param sanitizedIntent The sanitized {@link Intent} used for resolution. A sanitized Intent
* is an intent with potential PII removed from the original intent.
* Fields removed include extras and the host + path of the data, if
* defined.
* @param hostDigestPrefix The hash prefix of the instant app's domain.
* @param userHandle The user for which to resolve the instant app.
* @param token A unique identifier that will be provided in calls to {@link
* #onGetInstantAppIntentFilter(Intent, int[], UserHandle, String,
* InstantAppResolutionCallback)} and provided to the installer via {@link
* Intent#EXTRA_INSTANT_APP_TOKEN} to tie a single launch together.
* @param callback The {@link InstantAppResolutionCallback} to provide results to.
*
* @see InstantAppResolveInfo
*
* @deprecated Should implement {@link #onGetInstantAppResolveInfo(InstantAppRequestInfo,
* InstantAppResolutionCallback
*/
@Deprecated
public void onGetInstantAppResolveInfo(@NonNull Intent sanitizedIntent,
@Nullable int[] hostDigestPrefix, @NonNull UserHandle userHandle,
@NonNull String token, @NonNull InstantAppResolutionCallback callback) {
// If not overridden, forward to the old method.
onGetInstantAppResolveInfo(sanitizedIntent, hostDigestPrefix, token, callback);
}
/**
* Called to retrieve intent filters for potentially matching instant applications. Unlike
* {@link #onGetInstantAppResolveInfo(Intent, int[], UserHandle, String,
* InstantAppResolutionCallback)}, the response may take as long as necessary to respond. All
* {@link InstantAppResolveInfo}s provided in response to this method must be completely
* populated.
*
* @param sanitizedIntent The sanitized {@link Intent} used for resolution.
* @param hostDigestPrefix The hash prefix of the instant app's domain or null if no host is
* defined.
* @param userHandle The user for which to resolve the instant app.
* @param token A unique identifier that was provided in {@link #onGetInstantAppResolveInfo(
* Intent, int[], UserHandle, String, InstantAppResolutionCallback)} and provided
* to the currently visible installer via {@link Intent#EXTRA_INSTANT_APP_TOKEN}.
* @param callback The {@link InstantAppResolutionCallback} to provide results to.
*
* @deprecated Should implement {@link #onGetInstantAppIntentFilter(InstantAppRequestInfo,
* InstantAppResolutionCallback)}
*/
@Deprecated
public void onGetInstantAppIntentFilter(@NonNull Intent sanitizedIntent,
@Nullable int[] hostDigestPrefix, @NonNull UserHandle userHandle,
@NonNull String token, @NonNull InstantAppResolutionCallback callback) {
// If not overridden, forward to the old method.
onGetInstantAppIntentFilter(sanitizedIntent, hostDigestPrefix, token, callback);
}
/**
* Called to retrieve resolve info for instant applications immediately. The response will be
* ignored if not provided within a reasonable time. {@link InstantAppResolveInfo}s provided
* in response to this method may be partial to request a second phase of resolution which will
* result in a subsequent call to {@link #onGetInstantAppIntentFilter(InstantAppRequestInfo,
* InstantAppResolutionCallback)}
*
* @param request The parameters for this resolution request
* @param callback The {@link InstantAppResolutionCallback} to provide results to.
*
* @see InstantAppResolveInfo
*/
@MainThread
public void onGetInstantAppResolveInfo(@NonNull InstantAppRequestInfo request,
@NonNull InstantAppResolutionCallback callback) {
// If not overridden, forward to the old method.
onGetInstantAppResolveInfo(request.getIntent(), request.getHostDigestPrefix(),
request.getUserHandle(), request.getToken(), callback);
}
/**
* Called to retrieve intent filters for potentially matching instant applications. Unlike
* {@link #onGetInstantAppResolveInfo(InstantAppRequestInfo, InstantAppResolutionCallback)},
* the response may take as long as necessary to respond. All {@link InstantAppResolveInfo}s
* provided in response to this method must be completely populated.
*
* @param request The parameters for this resolution request
* @param callback The {@link InstantAppResolutionCallback} to provide results to.
*/
@MainThread
public void onGetInstantAppIntentFilter(@NonNull InstantAppRequestInfo request,
@NonNull InstantAppResolutionCallback callback) {
// If not overridden, forward to the old method.
onGetInstantAppIntentFilter(request.getIntent(), request.getHostDigestPrefix(),
request.getUserHandle(), request.getToken(), callback);
}
/**
* Returns a {@link Looper} to perform service operations on.
*/
Looper getLooper() {
return getBaseContext().getMainLooper();
}
@Override
public final void attachBaseContext(Context base) {
super.attachBaseContext(base);
mHandler = new ServiceHandler(getLooper());
}
@Override
public final IBinder onBind(Intent intent) {
return new IInstantAppResolver.Stub() {
@Override
public void getInstantAppResolveInfoList(InstantAppRequestInfo request, int sequence,
IRemoteCallback callback) {
if (DEBUG_INSTANT) {
Slog.v(TAG, "[" + request.getToken() + "] Phase1 called; posting");
}
final SomeArgs args = SomeArgs.obtain();
args.arg1 = request;
args.arg2 = callback;
mHandler.obtainMessage(ServiceHandler.MSG_GET_INSTANT_APP_RESOLVE_INFO, sequence,
0, args).sendToTarget();
}
@Override
public void getInstantAppIntentFilterList(InstantAppRequestInfo request,
IRemoteCallback callback) {
if (DEBUG_INSTANT) {
Slog.v(TAG, "[" + request.getToken() + "] Phase2 called; posting");
}
final SomeArgs args = SomeArgs.obtain();
args.arg1 = request;
args.arg2 = callback;
mHandler.obtainMessage(ServiceHandler.MSG_GET_INSTANT_APP_INTENT_FILTER, args)
.sendToTarget();
}
};
}
/**
* Callback to post results from instant app resolution.
*/
public static final class InstantAppResolutionCallback {
private final IRemoteCallback mCallback;
private final int mSequence;
/** @hide **/
@VisibleForTesting(visibility = VisibleForTesting.Visibility.PACKAGE)
public InstantAppResolutionCallback(int sequence, IRemoteCallback callback) {
mCallback = callback;
mSequence = sequence;
}
public void onInstantAppResolveInfo(List<InstantAppResolveInfo> resolveInfo) {
final Bundle data = new Bundle();
data.putParcelableList(EXTRA_RESOLVE_INFO, resolveInfo);
data.putInt(EXTRA_SEQUENCE, mSequence);
try {
mCallback.sendResult(data);
} catch (RemoteException e) {
}
}
}
private final class ServiceHandler extends Handler {
public static final int MSG_GET_INSTANT_APP_RESOLVE_INFO = 1;
public static final int MSG_GET_INSTANT_APP_INTENT_FILTER = 2;
public ServiceHandler(Looper looper) {
super(looper, null /*callback*/, true /*async*/);
}
@Override
@SuppressWarnings("unchecked")
public void handleMessage(Message message) {
final int action = message.what;
switch (action) {
case MSG_GET_INSTANT_APP_RESOLVE_INFO: {
final SomeArgs args = (SomeArgs) message.obj;
final InstantAppRequestInfo request = (InstantAppRequestInfo) args.arg1;
final IRemoteCallback callback = (IRemoteCallback) args.arg2;
args.recycle();
final int sequence = message.arg1;
if (DEBUG_INSTANT) {
Slog.d(TAG, "[" + request.getToken() + "] Phase1 request;"
+ " prefix: " + Arrays.toString(request.getHostDigestPrefix())
+ ", userId: " + request.getUserHandle().getIdentifier());
}
onGetInstantAppResolveInfo(request,
new InstantAppResolutionCallback(sequence, callback));
} break;
case MSG_GET_INSTANT_APP_INTENT_FILTER: {
final SomeArgs args = (SomeArgs) message.obj;
final InstantAppRequestInfo request = (InstantAppRequestInfo) args.arg1;
final IRemoteCallback callback = (IRemoteCallback) args.arg2;
args.recycle();
if (DEBUG_INSTANT) {
Slog.d(TAG, "[" + request.getToken() + "] Phase2 request;"
+ " prefix: " + Arrays.toString(request.getHostDigestPrefix())
+ ", userId: " + request.getUserHandle().getIdentifier());
}
onGetInstantAppIntentFilter(request,
new InstantAppResolutionCallback(-1 /*sequence*/, callback));
}
break;
default: {
throw new IllegalArgumentException("Unknown message: " + action);
}
}
}
}
}