Skip to content

End to End testing

Farees Hussain edited this page Aug 22, 2021 · 7 revisions

Overview

End-to-End tests test the app from an end user’s experience by simulating the real user scenario and validating the system under test and its components for integration and data integrity.

These tests play a major role in publishing the app. They run on a real or emulated device to make sure that our code interacts with the Android environment as expected,providing confidence in the final application or a feature when it's finished.

End-to-End tests in Oppia-android are written using UiAutomator. These tests are written under the instrumentation module and don’t have Gradle support.

instrumentation/ -- android test binaries for each test suite.
`-- src
    |-- java
    |   `-- org
    |       `-- oppia
    |           `-- android
    |               `-- instrumentation
    |                   `-- application -- Test application and modules
    `-- javatests
        `-- org
            `-- oppia
                `-- android
                    `-- instrumentation -- Test suites for each part of the app

These tests are run using Bazel and ADB. Each test suite tests a particular part of the app, each test suite has its own kt_android_library, android_binary and android_instrumentation_test.

Note: android_instrumentation_test target is not supported yet (#3617 for details).

How it works.

The android_binary of a test suite generates a test apk with the same name as the class of the test suite. This test apk is installed along with the original apk in the emulator. We use the adb am instrument command to run the test on the target device or emulator.

How to run an End-to-End test

Prerequisites:

  1. Setup Bazel for Oppia

  2. Add adb to the environment (platform-tools) i.e, add the following line to the .bashrc or the file path (note that this is platform dependent):

    export PATH=/home/<username>/Android/Sdk/platform-tools:$PATH
    
  3. download and install test-services-1.1.0.apk and orchestrator-1.1.0.apk in the emulator. (run command in the directory where both apk are downloaded)

    adb install -r test-services-1.1.0.apk && adb install -r orchestrator-1.1.0.apk
    
  4. java version 8 (Optional, only for uiautomatorviewer)

    java -version
    

    output: openjdk version "1.8.0_292" OpenJDK Runtime Environment (build 1.8.0_292-8u292-b10-0ubuntu1~18.04-b10) OpenJDK 64-Bit Server VM (build 25.292-b10, mixed mode)

Steps to run the tests

  1. Build the oppia_test and test suite android_binary from the instrumentation module
    bazel build :oppia_test && bazel build //instrumentation:<test suite classname>
    
    e.g.:
    bazel build :oppia_test && bazel build //instrumentation/src/javatests/org/oppia/android/instrumentation/player:ExplorationPlayerTestBinary
    
  2. install the oppia_test.apk and the test suite’s APK
    adb install -r bazel-bin/oppia_test.apk && adb install -r bazel-bin/instrumentation/<test suite classname>.apk
    
    e.g.:
    adb install -r bazel-bin/oppia_test.apk && adb install -r bazel-bin/instrumentation/src/javatests/org/oppia/android/instrumentation/player/ExplorationPlayerTestBinary.apk
    
  3. Run the instrumentation tests using am instrument command
    adb shell 'CLASSPATH=$(pm path androidx.test.services) app_process / \
    androidx.test.services.shellexecutor.ShellMain am instrument -w -e clearPackageData true \
    -e targetInstrumentation org.oppia.android.app.instrumentation/androidx.test.runner.AndroidJUnitRunner \
    androidx.test.orchestrator/.AndroidTestOrchestrator'
    
    Note: "-e clearPackageData true" is used to clear your app's data in between runs.

Best practices when writing End-to-End tests

  1. Each test suite should use the macro oppia_instrumentation_test which generates the necessary targets required for each test suite.
  2. Tests should follow the best practices for writing good tests in Oppia Android, including ensuring that tests are focused on a single, high-level behavior that the suite wishes to verify as working (such as being able to play fully through an exploration).
  3. Repetitive actions should be factored into helper methods. When these are generally useful for other end-to-end tests, they should be moved to EndToEndTestHelper.
  4. Prefer using existing helpers from EndToEndTestHelper rather than reimplementing them.
  5. Prefer testing whole end-to-end flows rather than specific behaviors. For example: opening the app, downloading a topic, and being able to play it is an excellent end-to-end test since it’s verifying multiple cross-stack behaviors. Conversely, testing that a thumbnail is aligned correctly on the home screen is less useful and ought to be tested in Robolectric or Espresso locally to the component showing that image.
  6. Use uiautomatorviewer to get details of each view such as resource-id, content description, class and other properties.