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 package android.contentcaptureservice.cts;
17 
18 import static com.android.compatibility.common.util.ShellUtils.runShellCommand;
19 
20 import android.app.Activity;
21 import android.contentcaptureservice.cts.CtsContentCaptureService.Session;
22 import android.os.Bundle;
23 import android.util.Log;
24 import android.view.View;
25 import android.view.contentcapture.ContentCaptureManager;
26 
27 import androidx.annotation.NonNull;
28 import androidx.annotation.Nullable;
29 
30 import java.util.concurrent.Callable;
31 import java.util.concurrent.CountDownLatch;
32 import java.util.concurrent.TimeUnit;
33 import java.util.concurrent.atomic.AtomicReference;
34 
35 /**
36  * Base class for all activities.
37  */
38 public abstract class AbstractContentCaptureActivity extends Activity {
39 
40     private final String mTag = getClass().getSimpleName();
41 
42     private int mRealTaskId;
43 
44     @Nullable
getContentCaptureManager()45     public ContentCaptureManager getContentCaptureManager() {
46         return getSystemService(ContentCaptureManager.class);
47     }
48 
49     @Override
onCreate(Bundle savedInstanceState)50     protected void onCreate(Bundle savedInstanceState) {
51         mRealTaskId = getTaskId();
52         Log.d(mTag, "onCreate(): taskId=" + mRealTaskId + ", decorView=" + getDecorView());
53 
54         super.onCreate(savedInstanceState);
55     }
56 
57     @Override
onStart()58     protected void onStart() {
59         Log.d(mTag, "onStart()");
60         super.onStart();
61     }
62 
63     @Override
onResume()64     protected void onResume() {
65         Log.d(mTag, "onResume(): decorViewId=" + getDecorView().getAutofillId());
66         super.onResume();
67     }
68 
69     @Override
onPause()70     protected void onPause() {
71         Log.d(mTag, "onPause()");
72         super.onPause();
73     }
74 
75     @Override
onStop()76     protected void onStop() {
77         Log.d(mTag, "onStop()");
78         super.onStop();
79     }
80 
81     @Override
onDestroy()82     protected void onDestroy() {
83         Log.d(mTag, "onDestroy()");
84         super.onDestroy();
85     }
86 
87     @NonNull
getDecorView()88     public final View getDecorView() {
89         return getWindow().getDecorView();
90     }
91 
92     /**
93      * Asserts the events generated when this session was launched and finished,
94      * without any custom / dynamic operations in between.
95      */
assertDefaultEvents(@onNull Session session)96     public abstract void assertDefaultEvents(@NonNull Session session);
97 
98     /**
99      * Gets the real task id associated with the activity, as {@link #getTaskId()} returns
100      * {@code -1} after it's gone.
101      */
getRealTaskId()102     public final int getRealTaskId() {
103         return mRealTaskId;
104     }
105 
106     /**
107      * Runs an action in the UI thread, and blocks caller until the action is finished.
108      */
syncRunOnUiThread(@onNull Runnable action)109     public final void syncRunOnUiThread(@NonNull Runnable action) {
110         syncRunOnUiThread(action, Helper.GENERIC_TIMEOUT_MS);
111     }
112 
113     /**
114      * Calls an action in the UI thread, and blocks caller until the action is finished.
115      */
syncCallOnUiThread(@onNull Callable<T> action)116     public final <T> T syncCallOnUiThread(@NonNull Callable<T> action) throws Exception {
117         final AtomicReference<T> result = new AtomicReference<>();
118         final AtomicReference<Exception> exception  = new AtomicReference<>();
119         syncRunOnUiThread(() -> {
120             try {
121                 result.set(action.call());
122             } catch (Exception e) {
123                 exception.set(e);
124             }
125         });
126         final Exception e = exception.get();
127         if (e != null) {
128             throw e;
129         }
130         return result.get();
131     }
132 
133     /**
134      * Run an action in the UI thread, and blocks caller until the action is finished or it times
135      * out.
136      */
syncRunOnUiThread(@onNull Runnable action, long timeoutMs)137     public final void syncRunOnUiThread(@NonNull Runnable action, long timeoutMs) {
138         final CountDownLatch latch = new CountDownLatch(1);
139         runOnUiThread(() -> {
140             action.run();
141             latch.countDown();
142         });
143         try {
144             if (!latch.await(timeoutMs, TimeUnit.MILLISECONDS)) {
145                 // TODO(b/120665995): throw RetryableException (once moved from Autofill to common)
146                 throw new IllegalStateException(
147                         String.format("action on UI thread timed out after %d ms", timeoutMs));
148             }
149         } catch (InterruptedException e) {
150             Thread.currentThread().interrupt();
151             throw new RuntimeException("Interrupted", e);
152         }
153     }
154 
155     /**
156      * Dumps the {@link ContentCaptureManager} state of the activity on logcat.
157      */
dumpIt()158     public void dumpIt() {
159         final String dump = runShellCommand(
160                 "dumpsys activity %s %s %s", getComponentName().flattenToString(),
161                 Activity.DUMP_ARG_DUMP_DUMPABLE, ContentCaptureManager.DUMPABLE_NAME);
162         Log.v(mTag, "dump it: " + dump);
163     }
164 }
165