1 /*
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 
17 package com.android.server.telecom;
18 
19 import android.net.Uri;
20 import android.os.Binder;
21 import android.os.Bundle;
22 import android.telecom.Log;
23 import android.telecom.PhoneAccountHandle;
24 
25 import com.android.internal.telecom.IInCallAdapter;
26 
27 import java.util.List;
28 
29 /**
30  * Receives call commands and updates from in-call app and passes them through to CallsManager.
31  * {@link InCallController} creates an instance of this class and passes it to the in-call app after
32  * binding to it. This adapter can receive commands and updates until the in-call app is unbound.
33  */
34 class InCallAdapter extends IInCallAdapter.Stub {
35     private final CallsManager mCallsManager;
36     private final CallIdMapper mCallIdMapper;
37     private final TelecomSystem.SyncRoot mLock;
38     private final String mOwnerPackageName;
39     private final String mOwnerPackageAbbreviation;
40 
41     /** Persists the specified parameters. */
42     public InCallAdapter(CallsManager callsManager, CallIdMapper callIdMapper,
43             TelecomSystem.SyncRoot lock, String ownerPackageName) {
44         mCallsManager = callsManager;
45         mCallIdMapper = callIdMapper;
46         mLock = lock;
47         mOwnerPackageName = ownerPackageName;
48         mOwnerPackageAbbreviation = Log.getPackageAbbreviation(ownerPackageName);
49     }
50 
51     @Override
52     public void answerCall(String callId, int videoState) {
53         try {
54             Log.startSession(LogUtils.Sessions.ICA_ANSWER_CALL, mOwnerPackageAbbreviation);
55             long token = Binder.clearCallingIdentity();
56             try {
57                 synchronized (mLock) {
58                     Log.d(this, "answerCall(%s,%d)", callId, videoState);
59                     Call call = mCallIdMapper.getCall(callId);
60                     if (call != null) {
61                         mCallsManager.answerCall(call, videoState);
62                     } else {
63                         Log.w(this, "answerCall, unknown call id: %s", callId);
64                     }
65                 }
66             } finally {
67                 Binder.restoreCallingIdentity(token);
68             }
69         } finally {
70             Log.endSession();
71         }
72     }
73 
74     @Override
75     public void deflectCall(String callId, Uri address) {
76         try {
77             Log.startSession(LogUtils.Sessions.ICA_DEFLECT_CALL, mOwnerPackageAbbreviation);
78             long token = Binder.clearCallingIdentity();
79             try {
80                 synchronized (mLock) {
81                     Log.i(this, "deflectCall - %s, %s ", callId, Log.pii(address));
82                     Call call = mCallIdMapper.getCall(callId);
83                     if (call != null) {
84                         mCallsManager.deflectCall(call, address);
85                     } else {
86                         Log.w(this, "deflectCall, unknown call id: %s", callId);
87                     }
88                 }
89             } finally {
90                 Binder.restoreCallingIdentity(token);
91             }
92         } finally {
93             Log.endSession();
94         }
95     }
96 
97     @Override
98     public void rejectCall(String callId, boolean rejectWithMessage, String textMessage) {
99         try {
100             Log.startSession(LogUtils.Sessions.ICA_REJECT_CALL, mOwnerPackageAbbreviation);
101 
102             int callingUid = Binder.getCallingUid();
103             long token = Binder.clearCallingIdentity();
104             try {
105                 synchronized (mLock) {
106                     // Check to make sure the in-call app's user isn't restricted from sending SMS.
107                     // If so, silently drop the outgoing message. Also drop message if the screen is
108                     // locked.
109                     if (!mCallsManager.isReplyWithSmsAllowed(callingUid)) {
110                         rejectWithMessage = false;
111                         textMessage = null;
112                     }
113 
114                     Log.d(this, "rejectCall(%s,%b,%s)", callId, rejectWithMessage, textMessage);
115                     Call call = mCallIdMapper.getCall(callId);
116                     if (call != null) {
117                         mCallsManager.rejectCall(call, rejectWithMessage, textMessage);
118                     } else {
119                         Log.w(this, "setRingback, unknown call id: %s", callId);
120                     }
121                 }
122             } finally {
123                 Binder.restoreCallingIdentity(token);
124             }
125         } finally {
126             Log.endSession();
127         }
128     }
129 
130     @Override
131     public void rejectCallWithReason(String callId,
132             @android.telecom.Call.RejectReason int rejectReason) {
133         try {
134             Log.startSession(LogUtils.Sessions.ICA_REJECT_CALL, mOwnerPackageAbbreviation);
135 
136             int callingUid = Binder.getCallingUid();
137             long token = Binder.clearCallingIdentity();
138             try {
139                 synchronized (mLock) {
140                     Log.d(this, "rejectCallWithReason(%s,%d)", callId, rejectReason);
141                     Call call = mCallIdMapper.getCall(callId);
142                     if (call != null) {
143                         mCallsManager.rejectCall(call, rejectReason);
144                     } else {
145                         Log.w(this, "rejectCallWithReason, unknown call id: %s", callId);
146                     }
147                 }
148             } finally {
149                 Binder.restoreCallingIdentity(token);
150             }
151         } finally {
152             Log.endSession();
153         }
154     }
155 
156     public void transferCall(String callId, Uri targetNumber, boolean isConfirmationRequired) {
157         try {
158             Log.startSession(LogUtils.Sessions.ICA_TRANSFER_CALL, mOwnerPackageAbbreviation);
159             long token = Binder.clearCallingIdentity();
160             try {
161                 synchronized (mLock) {
162                     Log.i(this, "transferCall - %s, %s, %b", callId, Log.pii(targetNumber),
163                             isConfirmationRequired);
164                     Call call = mCallIdMapper.getCall(callId);
165                     if (call != null) {
166                         mCallsManager.transferCall(call, targetNumber, isConfirmationRequired);
167                     } else {
168                         Log.w(this, "transferCall, unknown call id: %s", callId);
169                     }
170                 }
171             } finally {
172                 Binder.restoreCallingIdentity(token);
173             }
174         } finally {
175             Log.endSession();
176         }
177     }
178 
179     @Override
180     public void consultativeTransfer(String callId, String otherCallId) {
181         try {
182             Log.startSession(LogUtils.Sessions.ICA_CONSULTATIVE_TRANSFER,
183                     mOwnerPackageAbbreviation);
184             long token = Binder.clearCallingIdentity();
185             try {
186                 synchronized (mLock) {
187                     Log.i(this, "consultativeTransfer - %s, %s", callId, otherCallId);
188                     Call call = mCallIdMapper.getCall(callId);
189                     Call otherCall = mCallIdMapper.getCall(otherCallId);
190                     if (call != null && otherCall != null) {
191                         mCallsManager.transferCall(call, otherCall);
192                     } else {
193                         Log.w(this, "consultativeTransfer, unknown call id: %s or %s",
194                                 callId, otherCallId);
195                     }
196                 }
197             } finally {
198                 Binder.restoreCallingIdentity(token);
199             }
200         } finally {
201             Log.endSession();
202         }
203     }
204 
205     @Override
206     public void playDtmfTone(String callId, char digit) {
207         try {
208             Log.startSession("ICA.pDT", mOwnerPackageAbbreviation);
209             long token = Binder.clearCallingIdentity();
210             try {
211                 synchronized (mLock) {
212                     Log.d(this, "playDtmfTone(%s,%c)", callId, digit);
213                     Call call = mCallIdMapper.getCall(callId);
214                     if (call != null) {
215                         mCallsManager.playDtmfTone(call, digit);
216                     } else {
217                         Log.w(this, "playDtmfTone, unknown call id: %s", callId);
218                     }
219                 }
220             } finally {
221                 Binder.restoreCallingIdentity(token);
222             }
223         } finally {
224             Log.endSession();
225         }
226     }
227 
228     @Override
229     public void stopDtmfTone(String callId) {
230         try {
231             Log.startSession("ICA.sDT", mOwnerPackageAbbreviation);
232             long token = Binder.clearCallingIdentity();
233             try {
234                 synchronized (mLock) {
235                     Log.d(this, "stopDtmfTone(%s)", callId);
236                     Call call = mCallIdMapper.getCall(callId);
237                     if (call != null) {
238                         mCallsManager.stopDtmfTone(call);
239                     } else {
240                         Log.w(this, "stopDtmfTone, unknown call id: %s", callId);
241                     }
242                 }
243             } finally {
244                 Binder.restoreCallingIdentity(token);
245             }
246         } finally {
247             Log.endSession();
248         }
249     }
250 
251     @Override
252     public void postDialContinue(String callId, boolean proceed) {
253         try {
254             Log.startSession("ICA.pDC", mOwnerPackageAbbreviation);
255             long token = Binder.clearCallingIdentity();
256             try {
257                 synchronized (mLock) {
258                     Log.d(this, "postDialContinue(%s)", callId);
259                     Call call = mCallIdMapper.getCall(callId);
260                     if (call != null) {
261                         mCallsManager.postDialContinue(call, proceed);
262                     } else {
263                         Log.w(this, "postDialContinue, unknown call id: %s", callId);
264                     }
265                 }
266             } finally {
267                 Binder.restoreCallingIdentity(token);
268             }
269         } finally {
270             Log.endSession();
271         }
272     }
273 
274     @Override
275     public void disconnectCall(String callId) {
276         try {
277             Log.startSession(LogUtils.Sessions.ICA_DISCONNECT_CALL, mOwnerPackageAbbreviation);
278             long token = Binder.clearCallingIdentity();
279             try {
280                 synchronized (mLock) {
281                     Log.v(this, "disconnectCall: %s", callId);
282                     Call call = mCallIdMapper.getCall(callId);
283                     if (call != null) {
284                         mCallsManager.disconnectCall(call);
285                     } else {
286                         Log.w(this, "disconnectCall, unknown call id: %s", callId);
287                     }
288                 }
289             } finally {
290                 Binder.restoreCallingIdentity(token);
291             }
292         } finally {
293             Log.endSession();
294         }
295     }
296 
297     @Override
298     public void holdCall(String callId) {
299         try {
300             Log.startSession(LogUtils.Sessions.ICA_HOLD_CALL, mOwnerPackageAbbreviation);
301             long token = Binder.clearCallingIdentity();
302             try {
303                 synchronized (mLock) {
304                     Call call = mCallIdMapper.getCall(callId);
305                     if (call != null) {
306                         mCallsManager.holdCall(call);
307                     } else {
308                         Log.w(this, "holdCall, unknown call id: %s", callId);
309                     }
310                 }
311             } finally {
312                 Binder.restoreCallingIdentity(token);
313             }
314         } finally {
315             Log.endSession();
316         }
317     }
318 
319     @Override
320     public void unholdCall(String callId) {
321         try {
322             Log.startSession(LogUtils.Sessions.ICA_UNHOLD_CALL, mOwnerPackageAbbreviation);
323             long token = Binder.clearCallingIdentity();
324             try {
325                 synchronized (mLock) {
326                     Call call = mCallIdMapper.getCall(callId);
327                     if (call != null) {
328                         mCallsManager.unholdCall(call);
329                     } else {
330                         Log.w(this, "unholdCall, unknown call id: %s", callId);
331                     }
332                 }
333             } finally {
334                 Binder.restoreCallingIdentity(token);
335             }
336         } finally {
337             Log.endSession();
338         }
339     }
340 
341     @Override
342     public void phoneAccountSelected(String callId, PhoneAccountHandle accountHandle,
343             boolean setDefault) {
344         try {
345             Log.startSession("ICA.pAS", mOwnerPackageAbbreviation);
346             long token = Binder.clearCallingIdentity();
347             try {
348                 synchronized (mLock) {
349                     Call call = mCallIdMapper.getCall(callId);
350                     if (call != null) {
351                         mCallsManager.phoneAccountSelected(call, accountHandle, setDefault);
352                     } else {
353                         Log.w(this, "phoneAccountSelected, unknown call id: %s", callId);
354                     }
355                 }
356             } finally {
357                 Binder.restoreCallingIdentity(token);
358             }
359         } finally {
360             Log.endSession();
361         }
362     }
363 
364     @Override
365     public void mute(boolean shouldMute) {
366         try {
367             Log.startSession(LogUtils.Sessions.ICA_MUTE, mOwnerPackageAbbreviation);
368             long token = Binder.clearCallingIdentity();
369             try {
370                 synchronized (mLock) {
371                     mCallsManager.mute(shouldMute);
372                 }
373             } finally {
374                 Binder.restoreCallingIdentity(token);
375             }
376         } finally {
377             Log.endSession();
378         }
379     }
380 
381     @Override
382     public void setAudioRoute(int route, String bluetoothAddress) {
383         try {
384             Log.startSession(LogUtils.Sessions.ICA_SET_AUDIO_ROUTE, mOwnerPackageAbbreviation);
385             long token = Binder.clearCallingIdentity();
386             try {
387                 synchronized (mLock) {
388                     mCallsManager.setAudioRoute(route, bluetoothAddress);
389                 }
390             } finally {
391                 Binder.restoreCallingIdentity(token);
392             }
393         } finally {
394             Log.endSession();
395         }
396     }
397 
398     @Override
399     public void enterBackgroundAudioProcessing(String callId) {
400         try {
401             Log.startSession(LogUtils.Sessions.ICA_ENTER_AUDIO_PROCESSING,
402                     mOwnerPackageAbbreviation);
403             // TODO: enforce the extra permission.
404             Binder.withCleanCallingIdentity(() -> {
405                 synchronized (mLock) {
406                     Call call = mCallIdMapper.getCall(callId);
407                     if (call != null) {
408                         mCallsManager.enterBackgroundAudioProcessing(call, mOwnerPackageName);
409                     } else {
410                         Log.w(this, "enterBackgroundAudioProcessing, unknown call id: %s", callId);
411                     }
412                 }
413             });
414         } finally {
415             Log.endSession();
416         }
417     }
418 
419     @Override
420     public void exitBackgroundAudioProcessing(String callId, boolean shouldRing) {
421         try {
422             Log.startSession(LogUtils.Sessions.ICA_EXIT_AUDIO_PROCESSING,
423                     mOwnerPackageAbbreviation);
424             Binder.withCleanCallingIdentity(() -> {
425                 synchronized (mLock) {
426                     Call call = mCallIdMapper.getCall(callId);
427                     if (call != null) {
428                         mCallsManager.exitBackgroundAudioProcessing(call, shouldRing);
429                     } else {
430                         Log.w(InCallAdapter.this,
431                                 "exitBackgroundAudioProcessing, unknown call id: %s", callId);
432                     }
433                 }
434             });
435         } finally {
436             Log.endSession();
437         }
438     }
439 
440     @Override
441     public void conference(String callId, String otherCallId) {
442         try {
443             Log.startSession(LogUtils.Sessions.ICA_CONFERENCE, mOwnerPackageAbbreviation);
444             long token = Binder.clearCallingIdentity();
445             try {
446                 synchronized (mLock) {
447                     Call call = mCallIdMapper.getCall(callId);
448                     Call otherCall = mCallIdMapper.getCall(otherCallId);
449                     if (call != null && otherCall != null) {
450                         mCallsManager.conference(call, otherCall);
451                     } else {
452                         Log.w(this, "conference, unknown call id: %s or %s", callId, otherCallId);
453                     }
454                 }
455             } finally {
456                 Binder.restoreCallingIdentity(token);
457             }
458         } finally {
459             Log.endSession();
460         }
461     }
462 
463     @Override
464     public void splitFromConference(String callId) {
465         try {
466             Log.startSession("ICA.sFC", mOwnerPackageAbbreviation);
467             long token = Binder.clearCallingIdentity();
468             try {
469                 synchronized (mLock) {
470                     Call call = mCallIdMapper.getCall(callId);
471                     if (call != null) {
472                         call.splitFromConference();
473                     } else {
474                         Log.w(this, "splitFromConference, unknown call id: %s", callId);
475                     }
476                 }
477             } finally {
478                 Binder.restoreCallingIdentity(token);
479             }
480         } finally {
481             Log.endSession();
482         }
483     }
484 
485     @Override
486     public void mergeConference(String callId) {
487         try {
488             Log.startSession("ICA.mC", mOwnerPackageAbbreviation);
489             long token = Binder.clearCallingIdentity();
490             try {
491                 synchronized (mLock) {
492                     Call call = mCallIdMapper.getCall(callId);
493                     if (call != null) {
494                         call.mergeConference();
495                     } else {
496                         Log.w(this, "mergeConference, unknown call id: %s", callId);
497                     }
498                 }
499             } finally {
500                 Binder.restoreCallingIdentity(token);
501             }
502         } finally {
503             Log.endSession();
504         }
505     }
506 
507     @Override
508     public void swapConference(String callId) {
509         try {
510             Log.startSession("ICA.sC", mOwnerPackageAbbreviation);
511             long token = Binder.clearCallingIdentity();
512             try {
513                 synchronized (mLock) {
514                     Call call = mCallIdMapper.getCall(callId);
515                     if (call != null) {
516                         call.swapConference();
517                     } else {
518                         Log.w(this, "swapConference, unknown call id: %s", callId);
519                     }
520                 }
521             } finally {
522                 Binder.restoreCallingIdentity(token);
523             }
524         } finally {
525             Log.endSession();
526         }
527     }
528 
529     @Override
530     public void addConferenceParticipants(String callId, List<Uri> participants) {
531         try {
532             Log.startSession("ICA.aCP", mOwnerPackageAbbreviation);
533             long token = Binder.clearCallingIdentity();
534             try {
535                 synchronized (mLock) {
536                     Call call = mCallIdMapper.getCall(callId);
537                     if (call != null) {
538                         call.addConferenceParticipants(participants);
539                     } else {
540                         Log.w(this, "addConferenceParticipants, unknown call id: %s", callId);
541                     }
542                 }
543             } finally {
544                 Binder.restoreCallingIdentity(token);
545             }
546         } finally {
547             Log.endSession();
548         }
549     }
550 
551 
552     @Override
553     public void pullExternalCall(String callId) {
554         try {
555             Log.startSession("ICA.pEC", mOwnerPackageAbbreviation);
556             long token = Binder.clearCallingIdentity();
557             try {
558                 synchronized (mLock) {
559                     Call call = mCallIdMapper.getCall(callId);
560                     if (call != null) {
561                         call.pullExternalCall();
562                     } else {
563                         Log.w(this, "pullExternalCall, unknown call id: %s", callId);
564                     }
565                 }
566             } finally {
567                 Binder.restoreCallingIdentity(token);
568             }
569         } finally {
570             Log.endSession();
571         }
572     }
573 
574     @Override
575     public void sendCallEvent(String callId, String event, int targetSdkVer, Bundle extras) {
576         try {
577             Log.startSession("ICA.sCE", mOwnerPackageAbbreviation);
578             long token = Binder.clearCallingIdentity();
579             try {
580                 synchronized (mLock) {
581                     Call call = mCallIdMapper.getCall(callId);
582                     if (call != null) {
583                         call.sendCallEvent(event, targetSdkVer, extras);
584                     } else {
585                         Log.w(this, "sendCallEvent, unknown call id: %s", callId);
586                     }
587                 }
588             } finally {
589                 Binder.restoreCallingIdentity(token);
590             }
591         } finally {
592             Log.endSession();
593         }
594     }
595 
596     @Override
597     public void putExtras(String callId, Bundle extras) {
598         try {
599             Log.startSession("ICA.pE", mOwnerPackageAbbreviation);
600             long token = Binder.clearCallingIdentity();
601             try {
602                 synchronized (mLock) {
603                     Call call = mCallIdMapper.getCall(callId);
604                     if (call != null) {
605                         call.putExtras(Call.SOURCE_INCALL_SERVICE, extras);
606                     } else {
607                         Log.w(this, "putExtras, unknown call id: %s", callId);
608                     }
609                 }
610             } finally {
611                 Binder.restoreCallingIdentity(token);
612             }
613         } finally {
614             Log.endSession();
615         }
616     }
617 
618     @Override
619     public void removeExtras(String callId, List<String> keys) {
620         try {
621             Log.startSession("ICA.rE", mOwnerPackageAbbreviation);
622             long token = Binder.clearCallingIdentity();
623             try {
624                 synchronized (mLock) {
625                     Call call = mCallIdMapper.getCall(callId);
626                     if (call != null) {
627                         call.removeExtras(Call.SOURCE_INCALL_SERVICE, keys);
628                     } else {
629                         Log.w(this, "removeExtra, unknown call id: %s", callId);
630                     }
631                 }
632             } finally {
633                 Binder.restoreCallingIdentity(token);
634             }
635         } finally {
636             Log.endSession();
637         }
638     }
639 
640     @Override
641     public void turnOnProximitySensor() {
642         try {
643             Log.startSession("ICA.tOnPS", mOwnerPackageAbbreviation);
644             long token = Binder.clearCallingIdentity();
645             try {
646                 synchronized (mLock) {
647                     mCallsManager.turnOnProximitySensor();
648                 }
649             } finally {
650                 Binder.restoreCallingIdentity(token);
651             }
652         } finally {
653             Log.endSession();
654         }
655     }
656 
657     @Override
658     public void turnOffProximitySensor(boolean screenOnImmediately) {
659         try {
660             Log.startSession("ICA.tOffPS", mOwnerPackageAbbreviation);
661             long token = Binder.clearCallingIdentity();
662             try {
663                 synchronized (mLock) {
664                     mCallsManager.turnOffProximitySensor(screenOnImmediately);
665                 }
666             } finally {
667                 Binder.restoreCallingIdentity(token);
668             }
669         } finally {
670              Log.endSession();
671         }
672     }
673 
674     @Override
675     public void sendRttRequest(String callId) {
676         try {
677             Log.startSession("ICA.sRR");
678             long token = Binder.clearCallingIdentity();
679             try {
680                 synchronized (mLock) {
681                     Call call = mCallIdMapper.getCall(callId);
682                     if (call != null) {
683                         call.sendRttRequest();
684                     } else {
685                         Log.w(this, "stopRtt(): call %s not found", callId);
686                     }
687                 }
688             } finally {
689                 Binder.restoreCallingIdentity(token);
690             }
691         } finally {
692             Log.endSession();
693         }
694     }
695 
696     @Override
697     public void respondToRttRequest(String callId, int id, boolean accept) {
698         try {
699             Log.startSession("ICA.rTRR");
700             long token = Binder.clearCallingIdentity();
701             try {
702                 synchronized (mLock) {
703                     Call call = mCallIdMapper.getCall(callId);
704                     if (call != null) {
705                         call.handleRttRequestResponse(id, accept);
706                     } else {
707                         Log.w(this, "respondToRttRequest(): call %s not found", callId);
708                     }
709                 }
710             } finally {
711                 Binder.restoreCallingIdentity(token);
712             }
713         } finally {
714             Log.endSession();
715         }
716     }
717 
718     @Override
719     public void stopRtt(String callId) {
720         try {
721             Log.startSession("ICA.sRTT");
722             long token = Binder.clearCallingIdentity();
723             try {
724                 synchronized (mLock) {
725                     Call call = mCallIdMapper.getCall(callId);
726                     if (call != null) {
727                         call.stopRtt();
728                     } else {
729                         Log.w(this, "stopRtt(): call %s not found", callId);
730                     }
731                 }
732             } finally {
733                 Binder.restoreCallingIdentity(token);
734             }
735         } finally {
736             Log.endSession();
737         }
738     }
739 
740     @Override
741     public void setRttMode(String callId, int mode) {
742         try {
743             Log.startSession("ICA.sRM");
744             long token = Binder.clearCallingIdentity();
745             try {
746                 synchronized (mLock) {
747                     // TODO
748                 }
749             } finally {
750                 Binder.restoreCallingIdentity(token);
751             }
752         } finally {
753             Log.endSession();
754         }
755     }
756 
757     @Override
758     public void handoverTo(String callId, PhoneAccountHandle destAcct, int videoState,
759                            Bundle extras) {
760         try {
761             Log.startSession("ICA.hT", mOwnerPackageAbbreviation);
762             long token = Binder.clearCallingIdentity();
763             try {
764                 synchronized (mLock) {
765                     Call call = mCallIdMapper.getCall(callId);
766                     if (call != null) {
767                         call.handoverTo(destAcct, videoState, extras);
768                     } else {
769                         Log.w(this, "handoverTo, unknown call id: %s", callId);
770                     }
771                 }
772             } finally {
773                 Binder.restoreCallingIdentity(token);
774             }
775         } finally {
776             Log.endSession();
777         }
778     }
779 }
780