blob: d267f6ef9b9b3a248ac8e1db79b3cf81de65d213 [file] [log] [blame]
/*
* Copyright (C) 2022 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.adservices.adid;
import static android.adservices.common.AdServicesPermissions.ACCESS_ADSERVICES_AD_ID;
import android.adservices.common.AdServicesStatusUtils;
import android.adservices.common.CallerMetadata;
import android.adservices.common.SandboxedSdkContextUtils;
import android.annotation.CallbackExecutor;
import android.annotation.NonNull;
import android.annotation.RequiresPermission;
import android.app.sdksandbox.SandboxedSdkContext;
import android.content.Context;
import android.os.Build;
import android.os.LimitExceededException;
import android.os.OutcomeReceiver;
import android.os.RemoteException;
import android.os.SystemClock;
import androidx.annotation.RequiresApi;
import com.android.adservices.AdServicesCommon;
import com.android.adservices.LogUtil;
import com.android.adservices.ServiceBinder;
import java.util.Objects;
import java.util.concurrent.Executor;
/**
* AdId Manager provides APIs for app and ad-SDKs to access advertising ID. The advertising ID is a
* unique, per-device, user-resettable ID for advertising. It gives users better controls and
* provides developers with a simple, standard system to continue to monetize their apps via
* personalized ads (formerly known as interest-based ads).
*/
// TODO(b/269798827): Enable for R.
@RequiresApi(Build.VERSION_CODES.S)
public class AdIdManager {
/**
* Service used for registering AdIdManager in the system service registry.
*
* @hide
*/
public static final String ADID_SERVICE = "adid_service";
// When an app calls the AdId API directly, it sets the SDK name to empty string.
static final String EMPTY_SDK = "";
private Context mContext;
private ServiceBinder<IAdIdService> mServiceBinder;
/**
* Factory method for creating an instance of AdIdManager.
*
* @param context The {@link Context} to use
* @return A {@link AdIdManager} instance
*/
@NonNull
public static AdIdManager get(@NonNull Context context) {
// On T+, context.getSystemService() does more than just call constructor.
return (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU)
? context.getSystemService(AdIdManager.class)
: new AdIdManager(context);
}
/**
* Create AdIdManager
*
* @hide
*/
public AdIdManager(Context context) {
// In case the AdIdManager is initiated from inside a sdk_sandbox process the fields
// will be immediately rewritten by the initialize method below.
initialize(context);
}
/**
* Initializes {@link AdIdManager} with the given {@code context}.
*
* <p>This method is called by the {@link SandboxedSdkContext} to propagate the correct context.
* For more information check the javadoc on the {@link
* android.app.sdksandbox.SdkSandboxSystemServiceRegistry}.
*
* @hide
* @see android.app.sdksandbox.SdkSandboxSystemServiceRegistry
*/
public AdIdManager initialize(Context context) {
mContext = context;
mServiceBinder =
ServiceBinder.getServiceBinder(
context,
AdServicesCommon.ACTION_ADID_SERVICE,
IAdIdService.Stub::asInterface);
return this;
}
@NonNull
private IAdIdService getService() {
IAdIdService service = mServiceBinder.getService();
if (service == null) {
throw new IllegalStateException("Unable to find the service");
}
return service;
}
@NonNull
private Context getContext() {
return mContext;
}
/**
* Return the AdId.
*
* @param executor The executor to run callback.
* @param callback The callback that's called after adid are available or an error occurs.
* @throws SecurityException if caller is not authorized to call this API.
* @throws IllegalStateException if this API is not available.
* @throws LimitExceededException if rate limit was reached.
*/
@RequiresPermission(ACCESS_ADSERVICES_AD_ID)
@NonNull
public void getAdId(
@NonNull @CallbackExecutor Executor executor,
@NonNull OutcomeReceiver<AdId, Exception> callback) {
Objects.requireNonNull(executor);
Objects.requireNonNull(callback);
CallerMetadata callerMetadata =
new CallerMetadata.Builder()
.setBinderElapsedTimestamp(SystemClock.elapsedRealtime())
.build();
final IAdIdService service = getService();
String appPackageName = "";
String sdkPackageName = "";
// First check if context is SandboxedSdkContext or not
Context getAdIdRequestContext = getContext();
SandboxedSdkContext requestContext =
SandboxedSdkContextUtils.getAsSandboxedSdkContext(getAdIdRequestContext);
if (requestContext != null) {
sdkPackageName = requestContext.getSdkPackageName();
appPackageName = requestContext.getClientPackageName();
} else { // This is the case without the Sandbox.
appPackageName = getAdIdRequestContext.getPackageName();
}
try {
service.getAdId(
new GetAdIdParam.Builder()
.setAppPackageName(appPackageName)
.setSdkPackageName(sdkPackageName)
.build(),
callerMetadata,
new IGetAdIdCallback.Stub() {
@Override
public void onResult(GetAdIdResult resultParcel) {
executor.execute(
() -> {
if (resultParcel.isSuccess()) {
callback.onResult(
new AdId(
resultParcel.getAdId(),
resultParcel.isLatEnabled()));
} else {
callback.onError(
AdServicesStatusUtils.asException(
resultParcel));
}
});
}
@Override
public void onError(int resultCode) {
executor.execute(
() ->
callback.onError(
AdServicesStatusUtils.asException(resultCode)));
}
});
} catch (RemoteException e) {
LogUtil.e(e, "RemoteException");
callback.onError(e);
}
}
/**
* If the service is in an APK (as opposed to the system service), unbind it from the service to
* allow the APK process to die.
*
* @hide
*/
// TODO: change to @VisibleForTesting
public void unbindFromService() {
mServiceBinder.unbindFromService();
}
}