blob: e5db8cfa0989f958497757f6e4a1ccf3ccda67e6 [file] [log] [blame]
Ihab Awad542e0ea2014-05-16 10:22:16 -07001/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Tyler Gunnef9f6f92014-09-12 22:16:17 -070017package android.telecom;
Ihab Awad542e0ea2014-05-16 10:22:16 -070018
Brad Ebinger428cec92016-03-15 14:23:44 -070019import android.annotation.NonNull;
Tyler Gunnc7d10782019-04-11 11:41:15 -070020import android.annotation.Nullable;
Artur Satayev53ada2a2019-12-10 17:47:56 +000021import android.compat.annotation.UnsupportedAppUsage;
Ihab Awadc35ad022014-06-12 16:29:42 -070022import android.content.ComponentName;
Mathew Inwood8c854f82018-09-14 12:35:36 +010023import android.os.Build;
Ihab Awad542e0ea2014-05-16 10:22:16 -070024import android.os.Parcel;
25import android.os.Parcelable;
Evan Charlton134dd682014-11-25 14:12:57 -080026import android.os.Process;
27import android.os.UserHandle;
Ihab Awadc35ad022014-06-12 16:29:42 -070028
Ihab Awaddcaa5d62014-07-08 10:33:46 -070029import java.util.Objects;
Ihab Awad542e0ea2014-05-16 10:22:16 -070030
31/**
Santos Cordond9e614f2014-10-28 13:10:36 -070032 * The unique identifier for a {@link PhoneAccount}. A {@code PhoneAccountHandle} is made of two
33 * parts:
34 * <ul>
Brian Attwellad147f42014-12-19 11:37:16 -080035 * <li>The component name of the associated connection service.</li>
Santos Cordond9e614f2014-10-28 13:10:36 -070036 * <li>A string identifier that is unique across {@code PhoneAccountHandle}s with the same
Tyler Gunnc9bdcb42021-07-02 11:28:23 -070037 * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the
38 * {@link #getId()} provided does not expose personally identifying information. A
39 * {@link ConnectionService} should use an opaque token as the {@link PhoneAccountHandle}
40 * identifier.</li>
Santos Cordond9e614f2014-10-28 13:10:36 -070041 * </ul>
42 *
Brad Ebinger428cec92016-03-15 14:23:44 -070043 * Note: This Class requires a non-null {@link ComponentName} and {@link UserHandle} to operate
44 * properly. Passing in invalid parameters will generate a log warning.
45 *
Brian Attwellad147f42014-12-19 11:37:16 -080046 * See {@link PhoneAccount}, {@link TelecomManager}.
Ihab Awad542e0ea2014-05-16 10:22:16 -070047 */
Yorke Lee400470f2015-05-12 13:31:25 -070048public final class PhoneAccountHandle implements Parcelable {
Tyler Gunn8a979332022-05-06 12:31:17 -070049 /**
50 * Expected component name of Telephony phone accounts; ONLY used to determine if we should log
51 * the phone account handle ID.
52 */
53 private static final ComponentName TELEPHONY_COMPONENT_NAME =
54 new ComponentName("com.android.phone",
55 "com.android.services.telephony.TelephonyConnectionService");
56
Tyler Gunn17933eb2019-03-05 13:58:45 -080057 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 127403196)
Evan Charlton134dd682014-11-25 14:12:57 -080058 private final ComponentName mComponentName;
Mathew Inwood8c854f82018-09-14 12:35:36 +010059 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
Evan Charlton134dd682014-11-25 14:12:57 -080060 private final String mId;
61 private final UserHandle mUserHandle;
Ihab Awadc35ad022014-06-12 16:29:42 -070062
Tyler Gunnc9bdcb42021-07-02 11:28:23 -070063 /**
64 * Creates a new {@link PhoneAccountHandle}.
65 *
66 * @param componentName The {@link ComponentName} of the {@link ConnectionService} which
67 * services this {@link PhoneAccountHandle}.
68 * @param id A string identifier that is unique across {@code PhoneAccountHandle}s with the same
69 * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the
70 * ID provided does not expose personally identifying information. A
71 * {@link ConnectionService} should use an opaque token as the
72 * {@link PhoneAccountHandle} identifier.
Thomas Stuarte2b0dc52022-12-28 11:10:50 -080073 * <p>
74 * Note: Each String field is limited to 256 characters. This check is enforced when
75 * registering the PhoneAccount via
76 * {@link TelecomManager#registerPhoneAccount(PhoneAccount)} and will cause an
77 * {@link IllegalArgumentException} to be thrown if the character field limit is
78 * over 256.
Tyler Gunnc9bdcb42021-07-02 11:28:23 -070079 */
Evan Charlton6eb262c2014-07-19 18:18:19 -070080 public PhoneAccountHandle(
Brad Ebinger428cec92016-03-15 14:23:44 -070081 @NonNull ComponentName componentName,
82 @NonNull String id) {
Evan Charlton134dd682014-11-25 14:12:57 -080083 this(componentName, id, Process.myUserHandle());
84 }
85
Tyler Gunnc9bdcb42021-07-02 11:28:23 -070086 /**
87 * Creates a new {@link PhoneAccountHandle}.
88 *
89 * @param componentName The {@link ComponentName} of the {@link ConnectionService} which
90 * services this {@link PhoneAccountHandle}.
91 * @param id A string identifier that is unique across {@code PhoneAccountHandle}s with the same
92 * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the
93 * ID provided does not expose personally identifying information. A
94 * {@link ConnectionService} should use an opaque token as the
95 * {@link PhoneAccountHandle} identifier.
96 * @param userHandle The {@link UserHandle} associated with this {@link PhoneAccountHandle}.
Thomas Stuarte2b0dc52022-12-28 11:10:50 -080097 *
98 * <p>
99 * Note: Each String field is limited to 256 characters. This check is enforced when
100 * registering the PhoneAccount via
101 * {@link TelecomManager#registerPhoneAccount(PhoneAccount)} and will cause an
102 * {@link IllegalArgumentException} to be thrown if the character field limit is
103 * over 256.
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700104 */
Evan Charlton134dd682014-11-25 14:12:57 -0800105 public PhoneAccountHandle(
Brad Ebinger428cec92016-03-15 14:23:44 -0700106 @NonNull ComponentName componentName,
107 @NonNull String id,
108 @NonNull UserHandle userHandle) {
109 checkParameters(componentName, userHandle);
Ihab Awadc35ad022014-06-12 16:29:42 -0700110 mComponentName = componentName;
111 mId = id;
Evan Charlton134dd682014-11-25 14:12:57 -0800112 mUserHandle = userHandle;
Ihab Awadc35ad022014-06-12 16:29:42 -0700113 }
114
115 /**
Brian Attwellad147f42014-12-19 11:37:16 -0800116 * The {@code ComponentName} of the connection service which is responsible for making phone
117 * calls using this {@code PhoneAccountHandle}.
Ihab Awadc35ad022014-06-12 16:29:42 -0700118 *
119 * @return A suitable {@code ComponentName}.
120 */
121 public ComponentName getComponentName() {
122 return mComponentName;
123 }
124
125 /**
Ihab Awadb19a0bc2014-08-07 19:46:01 -0700126 * A string that uniquely distinguishes this particular {@code PhoneAccountHandle} from all the
Brian Attwellad147f42014-12-19 11:37:16 -0800127 * others supported by the connection service that created it.
Ihab Awadb19a0bc2014-08-07 19:46:01 -0700128 * <p>
Brian Attwellad147f42014-12-19 11:37:16 -0800129 * A connection service must select identifiers that are stable for the lifetime of
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700130 * their users' relationship with their service, across many Android devices. The identifier
131 * should be a stable opaque token which uniquely identifies the user within the service.
132 * Depending on how a service chooses to operate, a bad set of identifiers might be an
133 * increasing series of integers ({@code 0}, {@code 1}, {@code 2}, ...) that are generated
134 * locally on each phone and could collide with values generated on other phones or after a data
135 * wipe of a given phone.
136 * <p>
Santos Cordon8b338d42015-02-10 03:38:31 -0800137 * Important: A non-unique identifier could cause non-deterministic call-log backup/restore
138 * behavior.
139 *
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700140 * @return A service-specific unique opaque identifier for this {@code PhoneAccountHandle}.
Ihab Awadc35ad022014-06-12 16:29:42 -0700141 */
142 public String getId() {
143 return mId;
144 }
145
Evan Charlton134dd682014-11-25 14:12:57 -0800146 /**
147 * @return the {@link UserHandle} to use when connecting to this PhoneAccount.
Evan Charlton134dd682014-11-25 14:12:57 -0800148 */
149 public UserHandle getUserHandle() {
150 return mUserHandle;
151 }
152
Ihab Awad807fe0a2014-07-09 12:30:52 -0700153 @Override
154 public int hashCode() {
Evan Charlton134dd682014-11-25 14:12:57 -0800155 return Objects.hash(mComponentName, mId, mUserHandle);
Ihab Awadc35ad022014-06-12 16:29:42 -0700156 }
157
Santos Cordon98b27032014-07-14 03:32:56 -0700158 @Override
159 public String toString() {
Tyler Gunn8a979332022-05-06 12:31:17 -0700160 StringBuilder sb = new StringBuilder()
161 .append(mComponentName)
162 .append(", ");
163
164 if (TELEPHONY_COMPONENT_NAME.equals(mComponentName)) {
165 // Telephony phone account handles are now keyed by subscription id which is not
166 // sensitive.
167 sb.append(mId);
168 } else {
169 // Note: Log.pii called for mId as it can contain personally identifying phone account
170 // information such as SIP account IDs.
171 sb.append(Log.pii(mId));
172 }
173 sb.append(", ");
174 sb.append(mUserHandle);
175
176 return sb.toString();
Santos Cordon98b27032014-07-14 03:32:56 -0700177 }
178
Ihab Awad94cf4bf2014-07-17 11:21:19 -0700179 @Override
180 public boolean equals(Object other) {
Santos Cordon98b27032014-07-14 03:32:56 -0700181 return other != null &&
Evan Charlton6eb262c2014-07-19 18:18:19 -0700182 other instanceof PhoneAccountHandle &&
183 Objects.equals(((PhoneAccountHandle) other).getComponentName(),
184 getComponentName()) &&
Evan Charlton134dd682014-11-25 14:12:57 -0800185 Objects.equals(((PhoneAccountHandle) other).getId(), getId()) &&
186 Objects.equals(((PhoneAccountHandle) other).getUserHandle(), getUserHandle());
Santos Cordon98b27032014-07-14 03:32:56 -0700187 }
188
Ihab Awaddcaa5d62014-07-08 10:33:46 -0700189 //
Ihab Awad807fe0a2014-07-09 12:30:52 -0700190 // Parcelable implementation.
191 //
Ihab Awad542e0ea2014-05-16 10:22:16 -0700192
Sailesh Nepale7ef59a2014-07-08 21:48:22 -0700193 @Override
Ihab Awad542e0ea2014-05-16 10:22:16 -0700194 public int describeContents() {
195 return 0;
196 }
197
Sailesh Nepale7ef59a2014-07-08 21:48:22 -0700198 @Override
Ihab Awadc35ad022014-06-12 16:29:42 -0700199 public void writeToParcel(Parcel out, int flags) {
Evan Charlton134dd682014-11-25 14:12:57 -0800200 mComponentName.writeToParcel(out, flags);
Ihab Awadc35ad022014-06-12 16:29:42 -0700201 out.writeString(mId);
Evan Charlton134dd682014-11-25 14:12:57 -0800202 mUserHandle.writeToParcel(out, flags);
Ihab Awadc35ad022014-06-12 16:29:42 -0700203 }
Ihab Awad542e0ea2014-05-16 10:22:16 -0700204
Brad Ebinger428cec92016-03-15 14:23:44 -0700205 private void checkParameters(ComponentName componentName, UserHandle userHandle) {
206 if(componentName == null) {
207 android.util.Log.w("PhoneAccountHandle", new Exception("PhoneAccountHandle has " +
208 "been created with null ComponentName!"));
209 }
210 if(userHandle == null) {
211 android.util.Log.w("PhoneAccountHandle", new Exception("PhoneAccountHandle has " +
212 "been created with null UserHandle!"));
213 }
214 }
215
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700216 public static final @android.annotation.NonNull Creator<PhoneAccountHandle> CREATOR =
217 new Creator<PhoneAccountHandle>() {
Ihab Awad807fe0a2014-07-09 12:30:52 -0700218 @Override
Evan Charlton6eb262c2014-07-19 18:18:19 -0700219 public PhoneAccountHandle createFromParcel(Parcel in) {
220 return new PhoneAccountHandle(in);
Ihab Awad542e0ea2014-05-16 10:22:16 -0700221 }
222
Ihab Awad807fe0a2014-07-09 12:30:52 -0700223 @Override
Evan Charlton6eb262c2014-07-19 18:18:19 -0700224 public PhoneAccountHandle[] newArray(int size) {
225 return new PhoneAccountHandle[size];
Ihab Awad542e0ea2014-05-16 10:22:16 -0700226 }
227 };
228
Mathew Inwood31755f92018-12-20 13:53:36 +0000229 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
Evan Charlton6eb262c2014-07-19 18:18:19 -0700230 private PhoneAccountHandle(Parcel in) {
Evan Charlton134dd682014-11-25 14:12:57 -0800231 this(ComponentName.CREATOR.createFromParcel(in),
232 in.readString(),
233 UserHandle.CREATOR.createFromParcel(in));
Ihab Awadc35ad022014-06-12 16:29:42 -0700234 }
Tyler Gunnc7d10782019-04-11 11:41:15 -0700235
236 /**
237 * Determines if two {@link PhoneAccountHandle}s are from the same package.
238 *
239 * @param a Phone account handle to check for same {@link ConnectionService} package.
240 * @param b Other phone account handle to check for same {@link ConnectionService} package.
241 * @return {@code true} if the two {@link PhoneAccountHandle}s passed in belong to the same
242 * {@link ConnectionService} / package, {@code false} otherwise. Note: {@code null} phone
243 * account handles are considered equivalent to other {@code null} phone account handles.
244 * @hide
245 */
246 public static boolean areFromSamePackage(@Nullable PhoneAccountHandle a,
247 @Nullable PhoneAccountHandle b) {
248 String aPackageName = a != null ? a.getComponentName().getPackageName() : null;
249 String bPackageName = b != null ? b.getComponentName().getPackageName() : null;
250 return Objects.equals(aPackageName, bPackageName);
251 }
Ihab Awad542e0ea2014-05-16 10:22:16 -0700252}