A lightweight wrapper around Android's UI Automator library

Relax

Relax is a lightweight wrapper around Android’s UI Automator library. It helps write clear and concise UI tests:

Relax("com.example.myapp") {
    pressHome()
    launch()
    inputText("me@example.com", "id/email")
    inputText("v3ry_s3cur3", "id/password")
    click("Login")
    assertExists {
        textStartsWith("Welcome")
    }
}

Rather than:

val device = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation())
device.pressHome()
device.wait(Until.hasObject(By.pkg(device.launcherPackageName).depth(0)), 5_000)

val context = ApplicationProvider.getApplicationContext<Context>()
val packageName = "com.example.myapp"
val intent = context.packageManager.getLaunchIntentForPackage(packageName)!!
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK)
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
context.startActivity(intent)
device.wait(Until.hasObject(By.pkg(packageName).depth(0)), 5_000)

device.findObject(UiSelector().resourceId("$packageName:id/email")).text = "me@example.com"
device.findObject(UiSelector().resourceId("$packageName:id/password")).text = "v3ry_s3cur3"
device.findObject(UiSelector().text("Login")).click()
device.waitForIdle()

assertTrue(device.findObject(UiSelector().textStartsWith("Welcome")).exists())

Installation

In your project’s build.gradle:

dependencies {
    androidTestImplementation("com.emergetools.test:relax:0.1.0")
    // or use implementation if in test module
}

⚠️ Relax is currently in an experimental state and is subject to breaking changes.

Getting Started

Relax allows you to build UI test flows as a sequence of actions:

Relax("<your app's package name>") {
    // Arbitrary actions
}

Application Management

Launch the application:

launch()

Using a deeplink:

launchWithLink("custom://…")

Force-stop the application (requires com.android.test module):

forceStop()

Clear all data associated with the application package (requires com.android.test module):

clearData()

Click

By exact text:

click("Text")

By id:

click("id/my_view_id")

By class:

click(Button::class.java)

Comprehensive selector:

click {
    textContains("Milk")
    checkable(true)
    //
}

On specific coordinates:

click(x, y)

To long click simply use longClick instead of click.

Input Text

inputText("Ob-la-di", "Input Label")
inputText("ob-la-da", "id/input")
inputText("life goes on", EditText::class.java)
inputText("brah") {
    // Arbitrary selectors
}

Scroll

scrollForward("Text")
scrollBackward("id/list")
scrollToBeginning {
    // Arbitrary selectors
}
scrollToEnd(maxSwipes = 2, "id/list")

Swipe

swipeUp("Text")
swipeDown("id/carousel")
swipeLeft(CarouselView::class.java)
swipeRight {
    // Arbitrary selectors
}

System Buttons

pressHome()
pressBack()

Selectors

The same selectors as UiSelector are available:

click { // Or any other action that accepts selectors
    checkable(true)
    checked(true)
    child {
        // Arbitrary child selectors
    }
    className(MyView::class.java)
    className("com.example.myapp.MyView")
    classNameMatches("regex")
    description("description")
    descriptionContains("cript")
    descriptionMatches("regex")
    descriptionStartsWith("desc")
    enabled(true)
    focusable(true)
    focused(true)
    index(0)
    instance(0)
    longClickable(true)
    packageName("com.example.myapp")
    packageNameMatches("regex")
    parent {
        // Arbitrary parent selectors
    }
    resId("my_id")
    resIdMatches("regex")
    scrollable(true)
    selected(true)
    text("Text")
    textContains("x")
    textMatches("regex")
    textStartsWith("Te")
}

Assert

assertExists("Text")
assertExists("id/my_view")
assertNotExists(MyView::class.java)
assertNotExists {
    // Arbitrary selectors
}

Handle Errors

By default if an object cannot be found or an action cannot be completed the test will fail. This behavior can be easily modified.

Making some actions optional:

optional {
    inputText("SAVE10", "Coupon")
}
click("Purchase")

Ignoring all failures by default:

val config = FlowConfig(errorHandler = NoOpFlowErrorHandler)
Relax("id", config) {
    //
}

Custom error handler:

val errorHandler = object : FlowErrorHandler {
    override fun onError(error: Throwable) {
        //
    }
}
val config = FlowConfig(errorHandler = errorHandler)
Relax("id", config) {
    //
}

UI Automator Interoperability

Relax is a lightweight wrapper around UI Automator, therefore it’s easy to leverage the full feature set of UI Automator. For example:

Relax(/**/) {
    if (!device.isScreenOn) device.wakeUp()
    launch()
    click("Next")
    device.takeScreenshot(storePath)
    //
}

GitHub

View Github