blob: e3485deb90803583661e045ed56a04fae2215f8d [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 Gunn17933eb2019-03-05 13:58:45 -080049 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 127403196)
Evan Charlton134dd682014-11-25 14:12:57 -080050 private final ComponentName mComponentName;
Mathew Inwood8c854f82018-09-14 12:35:36 +010051 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
Evan Charlton134dd682014-11-25 14:12:57 -080052 private final String mId;
53 private final UserHandle mUserHandle;
Ihab Awadc35ad022014-06-12 16:29:42 -070054
Tyler Gunnc9bdcb42021-07-02 11:28:23 -070055 /**
56 * Creates a new {@link PhoneAccountHandle}.
57 *
58 * @param componentName The {@link ComponentName} of the {@link ConnectionService} which
59 * services this {@link PhoneAccountHandle}.
60 * @param id A string identifier that is unique across {@code PhoneAccountHandle}s with the same
61 * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the
62 * ID provided does not expose personally identifying information. A
63 * {@link ConnectionService} should use an opaque token as the
64 * {@link PhoneAccountHandle} identifier.
65 */
Evan Charlton6eb262c2014-07-19 18:18:19 -070066 public PhoneAccountHandle(
Brad Ebinger428cec92016-03-15 14:23:44 -070067 @NonNull ComponentName componentName,
68 @NonNull String id) {
Evan Charlton134dd682014-11-25 14:12:57 -080069 this(componentName, id, Process.myUserHandle());
70 }
71
Tyler Gunnc9bdcb42021-07-02 11:28:23 -070072 /**
73 * Creates a new {@link PhoneAccountHandle}.
74 *
75 * @param componentName The {@link ComponentName} of the {@link ConnectionService} which
76 * services this {@link PhoneAccountHandle}.
77 * @param id A string identifier that is unique across {@code PhoneAccountHandle}s with the same
78 * component name. Apps registering {@link PhoneAccountHandle}s should ensure that the
79 * ID provided does not expose personally identifying information. A
80 * {@link ConnectionService} should use an opaque token as the
81 * {@link PhoneAccountHandle} identifier.
82 * @param userHandle The {@link UserHandle} associated with this {@link PhoneAccountHandle}.
83 */
Evan Charlton134dd682014-11-25 14:12:57 -080084 public PhoneAccountHandle(
Brad Ebinger428cec92016-03-15 14:23:44 -070085 @NonNull ComponentName componentName,
86 @NonNull String id,
87 @NonNull UserHandle userHandle) {
88 checkParameters(componentName, userHandle);
Ihab Awadc35ad022014-06-12 16:29:42 -070089 mComponentName = componentName;
90 mId = id;
Evan Charlton134dd682014-11-25 14:12:57 -080091 mUserHandle = userHandle;
Ihab Awadc35ad022014-06-12 16:29:42 -070092 }
93
94 /**
Brian Attwellad147f42014-12-19 11:37:16 -080095 * The {@code ComponentName} of the connection service which is responsible for making phone
96 * calls using this {@code PhoneAccountHandle}.
Ihab Awadc35ad022014-06-12 16:29:42 -070097 *
98 * @return A suitable {@code ComponentName}.
99 */
100 public ComponentName getComponentName() {
101 return mComponentName;
102 }
103
104 /**
Ihab Awadb19a0bc2014-08-07 19:46:01 -0700105 * A string that uniquely distinguishes this particular {@code PhoneAccountHandle} from all the
Brian Attwellad147f42014-12-19 11:37:16 -0800106 * others supported by the connection service that created it.
Ihab Awadb19a0bc2014-08-07 19:46:01 -0700107 * <p>
Brian Attwellad147f42014-12-19 11:37:16 -0800108 * A connection service must select identifiers that are stable for the lifetime of
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700109 * their users' relationship with their service, across many Android devices. The identifier
110 * should be a stable opaque token which uniquely identifies the user within the service.
111 * Depending on how a service chooses to operate, a bad set of identifiers might be an
112 * increasing series of integers ({@code 0}, {@code 1}, {@code 2}, ...) that are generated
113 * locally on each phone and could collide with values generated on other phones or after a data
114 * wipe of a given phone.
115 * <p>
Santos Cordon8b338d42015-02-10 03:38:31 -0800116 * Important: A non-unique identifier could cause non-deterministic call-log backup/restore
117 * behavior.
118 *
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700119 * @return A service-specific unique opaque identifier for this {@code PhoneAccountHandle}.
Ihab Awadc35ad022014-06-12 16:29:42 -0700120 */
121 public String getId() {
122 return mId;
123 }
124
Evan Charlton134dd682014-11-25 14:12:57 -0800125 /**
126 * @return the {@link UserHandle} to use when connecting to this PhoneAccount.
Evan Charlton134dd682014-11-25 14:12:57 -0800127 */
128 public UserHandle getUserHandle() {
129 return mUserHandle;
130 }
131
Ihab Awad807fe0a2014-07-09 12:30:52 -0700132 @Override
133 public int hashCode() {
Evan Charlton134dd682014-11-25 14:12:57 -0800134 return Objects.hash(mComponentName, mId, mUserHandle);
Ihab Awadc35ad022014-06-12 16:29:42 -0700135 }
136
Santos Cordon98b27032014-07-14 03:32:56 -0700137 @Override
138 public String toString() {
Tyler Gunn76c01a52014-09-30 14:47:51 -0700139 // Note: Log.pii called for mId as it can contain personally identifying phone account
140 // information such as SIP account IDs.
Santos Cordon98b27032014-07-14 03:32:56 -0700141 return new StringBuilder().append(mComponentName)
142 .append(", ")
Tyler Gunn76c01a52014-09-30 14:47:51 -0700143 .append(Log.pii(mId))
Evan Charlton134dd682014-11-25 14:12:57 -0800144 .append(", ")
145 .append(mUserHandle)
Santos Cordon98b27032014-07-14 03:32:56 -0700146 .toString();
147 }
148
Ihab Awad94cf4bf2014-07-17 11:21:19 -0700149 @Override
150 public boolean equals(Object other) {
Santos Cordon98b27032014-07-14 03:32:56 -0700151 return other != null &&
Evan Charlton6eb262c2014-07-19 18:18:19 -0700152 other instanceof PhoneAccountHandle &&
153 Objects.equals(((PhoneAccountHandle) other).getComponentName(),
154 getComponentName()) &&
Evan Charlton134dd682014-11-25 14:12:57 -0800155 Objects.equals(((PhoneAccountHandle) other).getId(), getId()) &&
156 Objects.equals(((PhoneAccountHandle) other).getUserHandle(), getUserHandle());
Santos Cordon98b27032014-07-14 03:32:56 -0700157 }
158
Ihab Awaddcaa5d62014-07-08 10:33:46 -0700159 //
Ihab Awad807fe0a2014-07-09 12:30:52 -0700160 // Parcelable implementation.
161 //
Ihab Awad542e0ea2014-05-16 10:22:16 -0700162
Sailesh Nepale7ef59a2014-07-08 21:48:22 -0700163 @Override
Ihab Awad542e0ea2014-05-16 10:22:16 -0700164 public int describeContents() {
165 return 0;
166 }
167
Sailesh Nepale7ef59a2014-07-08 21:48:22 -0700168 @Override
Ihab Awadc35ad022014-06-12 16:29:42 -0700169 public void writeToParcel(Parcel out, int flags) {
Evan Charlton134dd682014-11-25 14:12:57 -0800170 mComponentName.writeToParcel(out, flags);
Ihab Awadc35ad022014-06-12 16:29:42 -0700171 out.writeString(mId);
Evan Charlton134dd682014-11-25 14:12:57 -0800172 mUserHandle.writeToParcel(out, flags);
Ihab Awadc35ad022014-06-12 16:29:42 -0700173 }
Ihab Awad542e0ea2014-05-16 10:22:16 -0700174
Brad Ebinger428cec92016-03-15 14:23:44 -0700175 private void checkParameters(ComponentName componentName, UserHandle userHandle) {
176 if(componentName == null) {
177 android.util.Log.w("PhoneAccountHandle", new Exception("PhoneAccountHandle has " +
178 "been created with null ComponentName!"));
179 }
180 if(userHandle == null) {
181 android.util.Log.w("PhoneAccountHandle", new Exception("PhoneAccountHandle has " +
182 "been created with null UserHandle!"));
183 }
184 }
185
Tyler Gunnc9bdcb42021-07-02 11:28:23 -0700186 public static final @android.annotation.NonNull Creator<PhoneAccountHandle> CREATOR =
187 new Creator<PhoneAccountHandle>() {
Ihab Awad807fe0a2014-07-09 12:30:52 -0700188 @Override
Evan Charlton6eb262c2014-07-19 18:18:19 -0700189 public PhoneAccountHandle createFromParcel(Parcel in) {
190 return new PhoneAccountHandle(in);
Ihab Awad542e0ea2014-05-16 10:22:16 -0700191 }
192
Ihab Awad807fe0a2014-07-09 12:30:52 -0700193 @Override
Evan Charlton6eb262c2014-07-19 18:18:19 -0700194 public PhoneAccountHandle[] newArray(int size) {
195 return new PhoneAccountHandle[size];
Ihab Awad542e0ea2014-05-16 10:22:16 -0700196 }
197 };
198
Mathew Inwood31755f92018-12-20 13:53:36 +0000199 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
Evan Charlton6eb262c2014-07-19 18:18:19 -0700200 private PhoneAccountHandle(Parcel in) {
Evan Charlton134dd682014-11-25 14:12:57 -0800201 this(ComponentName.CREATOR.createFromParcel(in),
202 in.readString(),
203 UserHandle.CREATOR.createFromParcel(in));
Ihab Awadc35ad022014-06-12 16:29:42 -0700204 }
Tyler Gunnc7d10782019-04-11 11:41:15 -0700205
206 /**
207 * Determines if two {@link PhoneAccountHandle}s are from the same package.
208 *
209 * @param a Phone account handle to check for same {@link ConnectionService} package.
210 * @param b Other phone account handle to check for same {@link ConnectionService} package.
211 * @return {@code true} if the two {@link PhoneAccountHandle}s passed in belong to the same
212 * {@link ConnectionService} / package, {@code false} otherwise. Note: {@code null} phone
213 * account handles are considered equivalent to other {@code null} phone account handles.
214 * @hide
215 */
216 public static boolean areFromSamePackage(@Nullable PhoneAccountHandle a,
217 @Nullable PhoneAccountHandle b) {
218 String aPackageName = a != null ? a.getComponentName().getPackageName() : null;
219 String bPackageName = b != null ? b.getComponentName().getPackageName() : null;
220 return Objects.equals(aPackageName, bPackageName);
221 }
Ihab Awad542e0ea2014-05-16 10:22:16 -0700222}