1 /*
2  * Copyright (C) 2018 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 
17 package com.android.server.telecom;
18 
19 import android.content.ComponentName;
20 import android.content.Context;
21 import android.content.Intent;
22 import android.content.ServiceConnection;
23 import android.content.pm.PackageManager;
24 import android.content.pm.ResolveInfo;
25 import android.content.pm.ServiceInfo;
26 import android.net.Uri;
27 import android.os.Handler;
28 import android.os.IBinder;
29 import android.os.RemoteException;
30 import android.telecom.Log;
31 import android.telecom.Logging.Session;
32 import android.telecom.PhoneAccountHandle;
33 import android.telecom.PhoneAccountSuggestion;
34 import android.telecom.PhoneAccountSuggestionService;
35 import android.telephony.PhoneNumberUtils;
36 import android.text.TextUtils;
37 
38 import com.android.internal.telecom.IPhoneAccountSuggestionCallback;
39 import com.android.internal.telecom.IPhoneAccountSuggestionService;
40 
41 import java.util.List;
42 import java.util.concurrent.CompletableFuture;
43 import java.util.stream.Collectors;
44 import java.util.stream.Stream;
45 
46 public class PhoneAccountSuggestionHelper {
47     private static final String TAG = PhoneAccountSuggestionHelper.class.getSimpleName();
48     private static ComponentName sOverrideComponent;
49 
50     /**
51      * @return A future (possible already complete) that contains a list of suggestions.
52      */
53     public static CompletableFuture<List<PhoneAccountSuggestion>>
54     bindAndGetSuggestions(Context context, Uri handle,
55             List<PhoneAccountHandle> availablePhoneAccounts) {
56         // Use the default list if there's no handle
57         if (handle == null) {
58             return CompletableFuture.completedFuture(getDefaultSuggestions(availablePhoneAccounts));
59         }
60         String number = PhoneNumberUtils.extractNetworkPortion(handle.getSchemeSpecificPart());
61 
62         // Use the default list if there's no service on the device.
63         ServiceInfo suggestionServiceInfo = getSuggestionServiceInfo(context);
64         if (suggestionServiceInfo == null) {
65             return CompletableFuture.completedFuture(getDefaultSuggestions(availablePhoneAccounts));
66         }
67 
68         Intent bindIntent = new Intent();
69         bindIntent.setComponent(new ComponentName(suggestionServiceInfo.packageName,
70                 suggestionServiceInfo.name));
71 
72         final CompletableFuture<List<PhoneAccountSuggestion>> future = new CompletableFuture<>();
73 
74         final Session logSession = Log.createSubsession();
75         ServiceConnection serviceConnection = new ServiceConnection() {
76             @Override
77             public void onServiceConnected(ComponentName name, IBinder _service) {
78                 Log.continueSession(logSession, "PASH.oSC");
79                 try {
80                     IPhoneAccountSuggestionService service =
81                             IPhoneAccountSuggestionService.Stub.asInterface(_service);
82                     // Set up the callback to complete the future once the remote side comes
83                     // back with suggestions
84                     IPhoneAccountSuggestionCallback callback =
85                             new IPhoneAccountSuggestionCallback.Stub() {
86                                 @Override
87                                 public void suggestPhoneAccounts(String suggestResultNumber,
88                                         List<PhoneAccountSuggestion> suggestions) {
89                                     if (TextUtils.equals(number, suggestResultNumber)) {
90                                         if (suggestions == null) {
91                                             future.complete(
92                                                     getDefaultSuggestions(availablePhoneAccounts));
93                                         } else {
94                                             future.complete(
95                                                     addDefaultsToProvidedSuggestions(
96                                                             suggestions, availablePhoneAccounts));
97                                         }
98                                     }
99                                 }
100                             };
101                     try {
102                         service.onAccountSuggestionRequest(callback, number);
103                     } catch (RemoteException e) {
104                         Log.w(TAG, "Cancelling suggestion process due to remote exception");
105                         future.complete(getDefaultSuggestions(availablePhoneAccounts));
106                     }
107                 } finally {
108                     Log.endSession();
109                 }
110             }
111 
112             @Override
113             public void onServiceDisconnected(ComponentName name) {
114                 // No locking needed -- CompletableFuture only lets one thread call complete.
115                 Log.continueSession(logSession, "PASH.oSD");
116                 try {
117                     if (!future.isDone()) {
118                         Log.w(TAG, "Cancelling suggestion process due to service disconnect");
119                     }
120                     future.complete(getDefaultSuggestions(availablePhoneAccounts));
121                 } finally {
122                     Log.endSession();
123                 }
124             }
125         };
126 
127         if (!context.bindService(bindIntent, serviceConnection, Context.BIND_AUTO_CREATE)) {
128             Log.i(TAG, "Cancelling suggestion process due to bind failure.");
129             future.complete(getDefaultSuggestions(availablePhoneAccounts));
130         }
131 
132         // Set up a timeout so that we're not waiting forever for the suggestion service.
133         Handler handler = new Handler();
134         handler.postDelayed(() -> {
135                     // No locking needed -- CompletableFuture only lets one thread call complete.
136                     Log.continueSession(logSession, "PASH.timeout");
137                     try {
138                         if (!future.isDone()) {
139                             Log.w(TAG, "Cancelling suggestion process due to timeout");
140                         }
141                         future.complete(getDefaultSuggestions(availablePhoneAccounts));
142                     } finally {
143                         Log.endSession();
144                     }
145                 },
146                 Timeouts.getPhoneAccountSuggestionServiceTimeout(context.getContentResolver()));
147         return future;
148     }
149 
150     private static List<PhoneAccountSuggestion> addDefaultsToProvidedSuggestions(
151             List<PhoneAccountSuggestion> providedSuggestions,
152             List<PhoneAccountHandle> availableAccountHandles) {
153         List<PhoneAccountHandle> handlesInSuggestions = providedSuggestions.stream()
154                 .map(PhoneAccountSuggestion::getPhoneAccountHandle)
155                 .collect(Collectors.toList());
156         List<PhoneAccountHandle> handlesToFillIn = availableAccountHandles.stream()
157                 .filter(handle -> !handlesInSuggestions.contains(handle))
158                 .collect(Collectors.toList());
159         List<PhoneAccountSuggestion> suggestionsToAppend = getDefaultSuggestions(handlesToFillIn);
160         return Stream.concat(suggestionsToAppend.stream(), providedSuggestions.stream())
161                 .collect( Collectors.toList());
162     }
163 
164     private static ServiceInfo getSuggestionServiceInfo(Context context) {
165         PackageManager packageManager = context.getPackageManager();
166         Intent queryIntent = new Intent();
167         queryIntent.setAction(PhoneAccountSuggestionService.SERVICE_INTERFACE);
168 
169         List<ResolveInfo> services;
170         if (sOverrideComponent == null) {
171             services = packageManager.queryIntentServices(queryIntent,
172                     PackageManager.MATCH_SYSTEM_ONLY);
173         } else {
174             Log.i(TAG, "Using override component %s", sOverrideComponent);
175             queryIntent.setComponent(sOverrideComponent);
176             services = packageManager.queryIntentServices(queryIntent,
177                     PackageManager.MATCH_ALL);
178         }
179 
180         if (services == null || services.size() == 0) {
181             Log.i(TAG, "No acct suggestion services found. Using defaults.");
182             return null;
183         }
184 
185         if (services.size() > 1) {
186             Log.w(TAG, "More than acct suggestion service found, cannot get unique service");
187             return null;
188         }
189         return services.get(0).serviceInfo;
190     }
191 
192     static void setOverrideServiceName(String flattenedComponentName) {
193         try {
194             sOverrideComponent = TextUtils.isEmpty(flattenedComponentName)
195                     ? null : ComponentName.unflattenFromString(flattenedComponentName);
196         } catch (Exception e) {
197             sOverrideComponent = null;
198             throw e;
199         }
200     }
201 
202     private static List<PhoneAccountSuggestion> getDefaultSuggestions(
203             List<PhoneAccountHandle> phoneAccountHandles) {
204         return phoneAccountHandles.stream().map(phoneAccountHandle ->
205                 new PhoneAccountSuggestion(phoneAccountHandle,
206                         PhoneAccountSuggestion.REASON_NONE, false)
207         ).collect(Collectors.toList());
208     }
209 }