Home | History | Annotate | Download | only in test
      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 
     17 package android.test;
     18 
     19 import android.app.Activity;
     20 import android.app.Instrumentation;
     21 import android.content.Intent;
     22 import android.os.Bundle;
     23 import android.util.Log;
     24 import android.view.KeyEvent;
     25 
     26 import java.lang.reflect.Field;
     27 import java.lang.reflect.InvocationTargetException;
     28 import java.lang.reflect.Method;
     29 import java.lang.reflect.Modifier;
     30 
     31 import junit.framework.TestCase;
     32 
     33 /**
     34  * A test case that has access to {@link Instrumentation}.
     35  */
     36 public 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 }