1/*
2 * Copyright (C) 2007 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
17package android.test;
18
19import android.app.Activity;
20import android.app.Instrumentation;
21import android.content.Intent;
22import android.os.Bundle;
23import android.util.Log;
24import android.view.KeyEvent;
25
26import java.lang.reflect.Field;
27import java.lang.reflect.InvocationTargetException;
28import java.lang.reflect.Method;
29import java.lang.reflect.Modifier;
30
31import junit.framework.TestCase;
32
33/**
34 * A test case that has access to {@link Instrumentation}.
35 */
36public class InstrumentationTestCase extends TestCase {
37
38    private Instrumentation mInstrumentation;
39
40    /**
41     * Injects instrumentation into this test case. This method is
42     * called by the test runner during test setup.
43     *
44     * @param instrumentation the instrumentation to use with this instance
45     */
46    public void injectInstrumentation(Instrumentation instrumentation) {
47        mInstrumentation = instrumentation;
48    }
49
50    /**
51     * Injects instrumentation into this test case. This method is
52     * called by the test runner during test setup.
53     *
54     * @param instrumentation the instrumentation to use with this instance
55     *
56     * @deprecated Incorrect spelling,
57     * use {@link #injectInstrumentation(android.app.Instrumentation)} instead.
58     */
59    @Deprecated
60    public void injectInsrumentation(Instrumentation instrumentation) {
61        injectInstrumentation(instrumentation);
62    }
63
64    /**
65     * Inheritors can access the instrumentation using this.
66     * @return instrumentation
67     */
68    public Instrumentation getInstrumentation() {
69        return mInstrumentation;
70    }
71
72    /**
73     * Utility method for launching an activity.
74     *
75     * <p>The {@link Intent} used to launch the Activity is:
76     *  action = {@link Intent#ACTION_MAIN}
77     *  extras = null, unless a custom bundle is provided here
78     * All other fields are null or empty.
79     *
80     * <p><b>NOTE:</b> The parameter <i>pkg</i> must refer to the package identifier of the
81     * package hosting the activity to be launched, which is specified in the AndroidManifest.xml
82     * file.  This is not necessarily the same as the java package name.
83     *
84     * @param pkg The package hosting the activity to be launched.
85     * @param activityCls The activity class to launch.
86     * @param extras Optional extra stuff to pass to the activity.
87     * @return The activity, or null if non launched.
88     */
89    public final <T extends Activity> T launchActivity(
90            String pkg,
91            Class<T> activityCls,
92            Bundle extras) {
93        Intent intent = new Intent(Intent.ACTION_MAIN);
94        if (extras != null) {
95            intent.putExtras(extras);
96        }
97        return launchActivityWithIntent(pkg, activityCls, intent);
98    }
99
100    /**
101     * Utility method for launching an activity with a specific Intent.
102     *
103     * <p><b>NOTE:</b> The parameter <i>pkg</i> must refer to the package identifier of the
104     * package hosting the activity to be launched, which is specified in the AndroidManifest.xml
105     * file.  This is not necessarily the same as the java package name.
106     *
107     * @param pkg The package hosting the activity to be launched.
108     * @param activityCls The activity class to launch.
109     * @param intent The intent to launch with
110     * @return The activity, or null if non launched.
111     */
112    @SuppressWarnings("unchecked")
113    public final <T extends Activity> T launchActivityWithIntent(
114            String pkg,
115            Class<T> activityCls,
116            Intent intent) {
117        intent.setClassName(pkg, activityCls.getName());
118        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
119        T activity = (T) getInstrumentation().startActivitySync(intent);
120        getInstrumentation().waitForIdleSync();
121        return activity;
122    }
123
124    /**
125     * Helper for running portions of a test on the UI thread.
126     *
127     * Note, in most cases it is simpler to annotate the test method with
128     * {@link android.test.UiThreadTest}, which will run the entire test method on the UI thread.
129     * Use this method if you need to switch in and out of the UI thread to perform your test.
130     *
131     * @param r runnable containing test code in the {@link Runnable#run()} method
132     */
133    public void runTestOnUiThread(final Runnable r) throws Throwable {
134        final Throwable[] exceptions = new Throwable[1];
135        getInstrumentation().runOnMainSync(new Runnable() {
136            public void run() {
137                try {
138                    r.run();
139                } catch (Throwable throwable) {
140                    exceptions[0] = throwable;
141                }
142            }
143        });
144        if (exceptions[0] != null) {
145            throw exceptions[0];
146        }
147    }
148
149    /**
150     * Runs the current unit test. If the unit test is annotated with
151     * {@link android.test.UiThreadTest}, the test is run on the UI thread.
152     */
153    @Override
154    protected void runTest() throws Throwable {
155        String fName = getName();
156        assertNotNull(fName);
157        Method method = null;
158        try {
159            // use getMethod to get all public inherited
160            // methods. getDeclaredMethods returns all
161            // methods of this class but excludes the
162            // inherited ones.
163            method = getClass().getMethod(fName, (Class[]) null);
164        } catch (NoSuchMethodException e) {
165            fail("Method \""+fName+"\" not found");
166        }
167
168        if (!Modifier.isPublic(method.getModifiers())) {
169            fail("Method \""+fName+"\" should be public");
170        }
171
172        int runCount = 1;
173        boolean isRepetitive = false;
174        if (method.isAnnotationPresent(FlakyTest.class)) {
175            runCount = method.getAnnotation(FlakyTest.class).tolerance();
176        } else if (method.isAnnotationPresent(RepetitiveTest.class)) {
177            runCount = method.getAnnotation(RepetitiveTest.class).numIterations();
178            isRepetitive = true;
179        }
180
181        if (method.isAnnotationPresent(UiThreadTest.class)) {
182            final int tolerance = runCount;
183            final boolean repetitive = isRepetitive;
184            final Method testMethod = method;
185            final Throwable[] exceptions = new Throwable[1];
186            getInstrumentation().runOnMainSync(new Runnable() {
187                public void run() {
188                    try {
189                        runMethod(testMethod, tolerance, repetitive);
190                    } catch (Throwable throwable) {
191                        exceptions[0] = throwable;
192                    }
193                }
194            });
195            if (exceptions[0] != null) {
196                throw exceptions[0];
197            }
198        } else {
199            runMethod(method, runCount, isRepetitive);
200        }
201    }
202
203    // For backwards-compatibility after adding isRepetitive
204    private void runMethod(Method runMethod, int tolerance) throws Throwable {
205        runMethod(runMethod, tolerance, false);
206    }
207
208    private void runMethod(Method runMethod, int tolerance, boolean isRepetitive) throws Throwable {
209        Throwable exception = null;
210
211        int runCount = 0;
212        do {
213            try {
214                runMethod.invoke(this, (Object[]) null);
215                exception = null;
216            } catch (InvocationTargetException e) {
217                e.fillInStackTrace();
218                exception = e.getTargetException();
219            } catch (IllegalAccessException e) {
220                e.fillInStackTrace();
221                exception = e;
222            } finally {
223                runCount++;
224                // Report current iteration number, if test is repetitive
225                if (isRepetitive) {
226                    Bundle iterations = new Bundle();
227                    iterations.putInt("currentiterations", runCount);
228                    getInstrumentation().sendStatus(2, iterations);
229                }
230            }
231        } while ((runCount < tolerance) && (isRepetitive || exception != null));
232
233        if (exception != null) {
234            throw exception;
235        }
236    }
237
238    /**
239     * Sends a series of key events through instrumentation and waits for idle. The sequence
240     * of keys is a string containing the key names as specified in KeyEvent, without the
241     * KEYCODE_ prefix. For instance: sendKeys("DPAD_LEFT A B C DPAD_CENTER"). Each key can
242     * be repeated by using the N* prefix. For instance, to send two KEYCODE_DPAD_LEFT, use
243     * the following: sendKeys("2*DPAD_LEFT").
244     *
245     * @param keysSequence The sequence of keys.
246     */
247    public void sendKeys(String keysSequence) {
248        final String[] keys = keysSequence.split(" ");
249        final int count = keys.length;
250
251        final Instrumentation instrumentation = getInstrumentation();
252
253        for (int i = 0; i < count; i++) {
254            String key = keys[i];
255            int repeater = key.indexOf('*');
256
257            int keyCount;
258            try {
259                keyCount = repeater == -1 ? 1 : Integer.parseInt(key.substring(0, repeater));
260            } catch (NumberFormatException e) {
261                Log.w("ActivityTestCase", "Invalid repeat count: " + key);
262                continue;
263            }
264
265            if (repeater != -1) {
266                key = key.substring(repeater + 1);
267            }
268
269            for (int j = 0; j < keyCount; j++) {
270                try {
271                    final Field keyCodeField = KeyEvent.class.getField("KEYCODE_" + key);
272                    final int keyCode = keyCodeField.getInt(null);
273                    try {
274                        instrumentation.sendKeyDownUpSync(keyCode);
275                    } catch (SecurityException e) {
276                        // Ignore security exceptions that are now thrown
277                        // when trying to send to another app, to retain
278                        // compatibility with existing tests.
279                    }
280                } catch (NoSuchFieldException e) {
281                    Log.w("ActivityTestCase", "Unknown keycode: KEYCODE_" + key);
282                    break;
283                } catch (IllegalAccessException e) {
284                    Log.w("ActivityTestCase", "Unknown keycode: KEYCODE_" + key);
285                    break;
286                }
287            }
288        }
289
290        instrumentation.waitForIdleSync();
291    }
292
293    /**
294     * Sends a series of key events through instrumentation and waits for idle. For instance:
295     * sendKeys(KEYCODE_DPAD_LEFT, KEYCODE_DPAD_CENTER).
296     *
297     * @param keys The series of key codes to send through instrumentation.
298     */
299    public void sendKeys(int... keys) {
300        final int count = keys.length;
301        final Instrumentation instrumentation = getInstrumentation();
302
303        for (int i = 0; i < count; i++) {
304            try {
305                instrumentation.sendKeyDownUpSync(keys[i]);
306            } catch (SecurityException e) {
307                // Ignore security exceptions that are now thrown
308                // when trying to send to another app, to retain
309                // compatibility with existing tests.
310            }
311        }
312
313        instrumentation.waitForIdleSync();
314    }
315
316    /**
317     * Sends a series of key events through instrumentation and waits for idle. Each key code
318     * must be preceded by the number of times the key code must be sent. For instance:
319     * sendRepeatedKeys(1, KEYCODE_DPAD_CENTER, 2, KEYCODE_DPAD_LEFT).
320     *
321     * @param keys The series of key repeats and codes to send through instrumentation.
322     */
323    public void sendRepeatedKeys(int... keys) {
324        final int count = keys.length;
325        if ((count & 0x1) == 0x1) {
326            throw new IllegalArgumentException("The size of the keys array must "
327                    + "be a multiple of 2");
328        }
329
330        final Instrumentation instrumentation = getInstrumentation();
331
332        for (int i = 0; i < count; i += 2) {
333            final int keyCount = keys[i];
334            final int keyCode = keys[i + 1];
335            for (int j = 0; j < keyCount; j++) {
336                try {
337                    instrumentation.sendKeyDownUpSync(keyCode);
338                } catch (SecurityException e) {
339                    // Ignore security exceptions that are now thrown
340                    // when trying to send to another app, to retain
341                    // compatibility with existing tests.
342                }
343            }
344        }
345
346        instrumentation.waitForIdleSync();
347    }
348
349    /**
350     * Make sure all resources are cleaned up and garbage collected before moving on to the next
351     * test. Subclasses that override this method should make sure they call super.tearDown()
352     * at the end of the overriding method.
353     *
354     * @throws Exception
355     */
356    @Override
357    protected void tearDown() throws Exception {
358        Runtime.getRuntime().gc();
359        Runtime.getRuntime().runFinalization();
360        Runtime.getRuntime().gc();
361        super.tearDown();
362    }
363}