| /* |
| * Copyright (C) 2014 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.telecom; |
| |
| import android.annotation.NonNull; |
| import android.annotation.Nullable; |
| import android.compat.annotation.UnsupportedAppUsage; |
| import android.content.ComponentName; |
| import android.os.Build; |
| import android.os.Parcel; |
| import android.os.Parcelable; |
| import android.os.Process; |
| import android.os.UserHandle; |
| |
| import java.util.Objects; |
| |
| /** |
| * The unique identifier for a {@link PhoneAccount}. A {@code PhoneAccountHandle} is made of two |
| * parts: |
| * <ul> |
| * <li>The component name of the associated connection service.</li> |
| * <li>A string identifier that is unique across {@code PhoneAccountHandle}s with the same |
| * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the |
| * {@link #getId()} provided does not expose personally identifying information. A |
| * {@link ConnectionService} should use an opaque token as the {@link PhoneAccountHandle} |
| * identifier.</li> |
| * </ul> |
| * |
| * Note: This Class requires a non-null {@link ComponentName} and {@link UserHandle} to operate |
| * properly. Passing in invalid parameters will generate a log warning. |
| * |
| * See {@link PhoneAccount}, {@link TelecomManager}. |
| */ |
| public final class PhoneAccountHandle implements Parcelable { |
| /** |
| * Expected component name of Telephony phone accounts; ONLY used to determine if we should log |
| * the phone account handle ID. |
| */ |
| private static final ComponentName TELEPHONY_COMPONENT_NAME = |
| new ComponentName("com.android.phone", |
| "com.android.services.telephony.TelephonyConnectionService"); |
| |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 127403196) |
| private final ComponentName mComponentName; |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) |
| private final String mId; |
| private final UserHandle mUserHandle; |
| |
| /** |
| * Creates a new {@link PhoneAccountHandle}. |
| * |
| * @param componentName The {@link ComponentName} of the {@link ConnectionService} which |
| * services this {@link PhoneAccountHandle}. |
| * @param id A string identifier that is unique across {@code PhoneAccountHandle}s with the same |
| * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the |
| * ID provided does not expose personally identifying information. A |
| * {@link ConnectionService} should use an opaque token as the |
| * {@link PhoneAccountHandle} identifier. |
| * <p> |
| * Note: Each String field is limited to 256 characters. This check is enforced when |
| * registering the PhoneAccount via |
| * {@link TelecomManager#registerPhoneAccount(PhoneAccount)} and will cause an |
| * {@link IllegalArgumentException} to be thrown if the character field limit is |
| * over 256. |
| */ |
| public PhoneAccountHandle( |
| @NonNull ComponentName componentName, |
| @NonNull String id) { |
| this(componentName, id, Process.myUserHandle()); |
| } |
| |
| /** |
| * Creates a new {@link PhoneAccountHandle}. |
| * |
| * @param componentName The {@link ComponentName} of the {@link ConnectionService} which |
| * services this {@link PhoneAccountHandle}. |
| * @param id A string identifier that is unique across {@code PhoneAccountHandle}s with the same |
| * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the |
| * ID provided does not expose personally identifying information. A |
| * {@link ConnectionService} should use an opaque token as the |
| * {@link PhoneAccountHandle} identifier. |
| * @param userHandle The {@link UserHandle} associated with this {@link PhoneAccountHandle}. |
| * |
| * <p> |
| * Note: Each String field is limited to 256 characters. This check is enforced when |
| * registering the PhoneAccount via |
| * {@link TelecomManager#registerPhoneAccount(PhoneAccount)} and will cause an |
| * {@link IllegalArgumentException} to be thrown if the character field limit is |
| * over 256. |
| */ |
| public PhoneAccountHandle( |
| @NonNull ComponentName componentName, |
| @NonNull String id, |
| @NonNull UserHandle userHandle) { |
| checkParameters(componentName, userHandle); |
| mComponentName = componentName; |
| mId = id; |
| mUserHandle = userHandle; |
| } |
| |
| /** |
| * The {@code ComponentName} of the connection service which is responsible for making phone |
| * calls using this {@code PhoneAccountHandle}. |
| * |
| * @return A suitable {@code ComponentName}. |
| */ |
| public ComponentName getComponentName() { |
| return mComponentName; |
| } |
| |
| /** |
| * A string that uniquely distinguishes this particular {@code PhoneAccountHandle} from all the |
| * others supported by the connection service that created it. |
| * <p> |
| * A connection service must select identifiers that are stable for the lifetime of |
| * their users' relationship with their service, across many Android devices. The identifier |
| * should be a stable opaque token which uniquely identifies the user within the service. |
| * Depending on how a service chooses to operate, a bad set of identifiers might be an |
| * increasing series of integers ({@code 0}, {@code 1}, {@code 2}, ...) that are generated |
| * locally on each phone and could collide with values generated on other phones or after a data |
| * wipe of a given phone. |
| * <p> |
| * Important: A non-unique identifier could cause non-deterministic call-log backup/restore |
| * behavior. |
| * |
| * @return A service-specific unique opaque identifier for this {@code PhoneAccountHandle}. |
| */ |
| public String getId() { |
| return mId; |
| } |
| |
| /** |
| * @return the {@link UserHandle} to use when connecting to this PhoneAccount. |
| */ |
| public UserHandle getUserHandle() { |
| return mUserHandle; |
| } |
| |
| @Override |
| public int hashCode() { |
| return Objects.hash(mComponentName, mId, mUserHandle); |
| } |
| |
| @Override |
| public String toString() { |
| StringBuilder sb = new StringBuilder() |
| .append(mComponentName) |
| .append(", "); |
| |
| if (TELEPHONY_COMPONENT_NAME.equals(mComponentName)) { |
| // Telephony phone account handles are now keyed by subscription id which is not |
| // sensitive. |
| sb.append(mId); |
| } else { |
| // Note: Log.pii called for mId as it can contain personally identifying phone account |
| // information such as SIP account IDs. |
| sb.append(Log.pii(mId)); |
| } |
| sb.append(", "); |
| sb.append(mUserHandle); |
| |
| return sb.toString(); |
| } |
| |
| @Override |
| public boolean equals(Object other) { |
| return other != null && |
| other instanceof PhoneAccountHandle && |
| Objects.equals(((PhoneAccountHandle) other).getComponentName(), |
| getComponentName()) && |
| Objects.equals(((PhoneAccountHandle) other).getId(), getId()) && |
| Objects.equals(((PhoneAccountHandle) other).getUserHandle(), getUserHandle()); |
| } |
| |
| // |
| // Parcelable implementation. |
| // |
| |
| @Override |
| public int describeContents() { |
| return 0; |
| } |
| |
| @Override |
| public void writeToParcel(Parcel out, int flags) { |
| mComponentName.writeToParcel(out, flags); |
| out.writeString(mId); |
| mUserHandle.writeToParcel(out, flags); |
| } |
| |
| private void checkParameters(ComponentName componentName, UserHandle userHandle) { |
| if(componentName == null) { |
| android.util.Log.w("PhoneAccountHandle", new Exception("PhoneAccountHandle has " + |
| "been created with null ComponentName!")); |
| } |
| if(userHandle == null) { |
| android.util.Log.w("PhoneAccountHandle", new Exception("PhoneAccountHandle has " + |
| "been created with null UserHandle!")); |
| } |
| } |
| |
| public static final @android.annotation.NonNull Creator<PhoneAccountHandle> CREATOR = |
| new Creator<PhoneAccountHandle>() { |
| @Override |
| public PhoneAccountHandle createFromParcel(Parcel in) { |
| return new PhoneAccountHandle(in); |
| } |
| |
| @Override |
| public PhoneAccountHandle[] newArray(int size) { |
| return new PhoneAccountHandle[size]; |
| } |
| }; |
| |
| @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) |
| private PhoneAccountHandle(Parcel in) { |
| this(ComponentName.CREATOR.createFromParcel(in), |
| in.readString(), |
| UserHandle.CREATOR.createFromParcel(in)); |
| } |
| |
| /** |
| * Determines if two {@link PhoneAccountHandle}s are from the same package. |
| * |
| * @param a Phone account handle to check for same {@link ConnectionService} package. |
| * @param b Other phone account handle to check for same {@link ConnectionService} package. |
| * @return {@code true} if the two {@link PhoneAccountHandle}s passed in belong to the same |
| * {@link ConnectionService} / package, {@code false} otherwise. Note: {@code null} phone |
| * account handles are considered equivalent to other {@code null} phone account handles. |
| * @hide |
| */ |
| public static boolean areFromSamePackage(@Nullable PhoneAccountHandle a, |
| @Nullable PhoneAccountHandle b) { |
| String aPackageName = a != null ? a.getComponentName().getPackageName() : null; |
| String bPackageName = b != null ? b.getComponentName().getPackageName() : null; |
| return Objects.equals(aPackageName, bPackageName); |
| } |
| } |