9.5 KiB
Ravenwood for Test Authors
The Ravenwood testing environment runs inside a single Java process on the host side, and provides a limited yet growing set of Android API functionality.
Ravenwood explicitly does not support “large” integration tests that expect a fully booted Android OS. Instead, it’s more suited for “small” and “medium” tests where your code-under-test has been factored to remove dependencies on a fully booted device.
When writing tests under Ravenwood, all Android API symbols associated with your declared sdk_version
are available to link against using, but unsupported APIs will throw an exception. This design choice enables mocking of unsupported APIs, and supports sharing of test code to build “bivalent” test suites that run against either Ravenwood or a traditional device.
Manually running tests
To run all Ravenwood tests, use:
./frameworks/base/ravenwood/scripts/run-ravenwood-tests.sh
To run a specific test, use "atest" as normal, selecting the test from a Ravenwood suite such as:
atest CtsOsTestCasesRavenwood:ParcelTest\#testSetDataCapacityNegative
Typical test structure
Below are the typical steps needed to add a straightforward “small” unit test:
- Define an
android_ravenwood_test
rule in yourAndroid.bp
file:
android_ravenwood_test {
name: "MyTestsRavenwood",
static_libs: [
"androidx.annotation_annotation",
"androidx.test.ext.junit",
"androidx.test.rules",
],
srcs: [
"src/com/example/MyCode.java",
"tests/src/com/example/MyCodeTest.java",
],
sdk_version: "test_current",
auto_gen_config: true,
}
- Write your unit test just like you would for an Android device:
import android.platform.test.annotations.DisabledOnRavenwood;
import android.platform.test.ravenwood.RavenwoodRule;
import androidx.test.ext.junit.runners.AndroidJUnit4;
import org.junit.Test;
import org.junit.runner.RunWith;
@RunWith(AndroidJUnit4.class)
public class MyCodeTest {
@Test
public void testSimple() {
// ...
}
}
- APIs available under Ravenwood are stateless by default. If your test requires explicit states (such as defining the UID you’re running under, or requiring a main
Looper
thread), add aRavenwoodRule
to declare that:
import android.platform.test.annotations.DisabledOnRavenwood;
import android.platform.test.ravenwood.RavenwoodRule;
import androidx.test.runner.AndroidJUnit4;
import org.junit.Test;
import org.junit.runner.RunWith;
@RunWith(AndroidJUnit4.class)
public class MyCodeTest {
@Rule
public final RavenwoodRule mRavenwood = new RavenwoodRule.Builder()
.setProcessApp()
.setProvideMainThread(true)
.build();
Once you’ve defined your test, you can use typical commands to execute it locally:
$ atest --host MyTestsRavenwood
Note: There's a known bug #312525698 where
atest
currently requires a connected device to run Ravenwood tests, but that device isn't used for testing. Using the--host
argument above is a way to bypass this requirement until the bug is fixed.
You can also run your new tests automatically via TEST_MAPPING
rules like this:
{
"ravenwood-presubmit": [
{
"name": "MyTestsRavenwood",
"host": true
}
]
}
Note: There's a known bug #308854804 where
TEST_MAPPING
is not being applied, so we're currently planning to run all Ravenwood tests unconditionally in presubmit for changes toframeworks/base/
andcts/
until there is a better path forward.
Strategies for feature flags
Ravenwood supports writing tests against logic that uses feature flags through the existing SetFlagsRule
infrastructure maintained by the feature flagging team:
import android.platform.test.flag.junit.SetFlagsRule;
@RunWith(AndroidJUnit4.class)
public class MyCodeTest {
@Rule
public final SetFlagsRule mSetFlagsRule = new SetFlagsRule(SetFlagsRule.DefaultInitValueType.NULL_DEFAULT);
@Test
public void testEnabled() {
mSetFlagsRule.enableFlags(Flags.FLAG_MY_FLAG);
// verify test logic that depends on flag being enabled
}
This naturally composes together well with any RavenwoodRule
that your test may need.
While SetFlagsRule
is generally a best-practice (as it can explicitly confirm behaviors for both "on" and "off" states), you may need to write tests that use CheckFlagsRule
(such as when writing CTS). Ravenwood currently supports CheckFlagsRule
by offering "all-on" and "all-off" behaviors:
import android.platform.test.flag.junit.CheckFlagsRule;
import android.platform.test.flag.junit.DeviceFlagsValueProvider;
import android.platform.test.flag.junit.RavenwoodFlagsValueProvider;
import android.platform.test.ravenwood.RavenwoodRule;
@RunWith(AndroidJUnit4.class)
public class MyCodeTest {
@Rule
public final CheckFlagsRule mCheckFlagsRule = RavenwoodRule.isUnderRavenwood()
? RavenwoodFlagsValueProvider.createAllOnCheckFlagsRule()
: DeviceFlagsValueProvider.createCheckFlagsRule();
Ravenwood currently doesn't have knowledge of the "default" value of any flags, so using createAllOnCheckFlagsRule()
is recommended to verify the widest possible set of behaviors. The example code above falls back to using default values from DeviceFlagsValueProvider
when not running on Ravenwood.
Strategies for migration/bivalent tests
Ravenwood aims to support tests that are written in a “bivalent” way, where the same test code can be dual-compiled to run on both a real Android device and under a Ravenwood environment.
In situations where a test method depends on API functionality not yet available under Ravenwood, we provide an annotation to quietly “ignore” that test under Ravenwood, while continuing to validate that test on real devices. The annotation can be applied to either individual methods or to an entire test class. Please note that your test class must declare a RavenwoodRule
for the annotation to take effect.
Test authors are encouraged to provide a blockedBy
or reason
argument to help future maintainers understand why a test is being ignored, and under what conditions it might be supported in the future.
@RunWith(AndroidJUnit4.class)
public class MyCodeTest {
@Rule
public final RavenwoodRule mRavenwood = new RavenwoodRule();
@Test
public void testSimple() {
// Simple test that runs on both devices and Ravenwood
}
@Test
@DisabledOnRavenwood(blockedBy = PackageManager.class)
public void testComplex() {
// Complex test that runs on devices, but is ignored under Ravenwood
}
}
At the moment, the android.content.res.Resources
subsystem isn't yet supported under Ravenwood, but you may still want to dual-compile test suites that depend on references to resources. Below is a strategy for supporting dual-compiliation, where you can "borrow" the generated resource symbols from your traditional android_test
target:
android_test {
name: "MyTestsDevice",
resource_dirs: ["res"],
...
android_ravenwood_test {
name: "MyTestsRavenwood",
srcs: [
":MyTestsDevice{.aapt.srcjar}",
...
Strategies for unsupported APIs
As you write tests against Ravenwood, you’ll likely discover API dependencies that aren’t supported yet. Here’s a few strategies that can help you make progress:
- Your code-under-test may benefit from subtle dependency refactoring to reduce coupling. (For example, providing a specific
File
argument instead of deriving paths internally from aContext
orEnvironment
.)- One common use-case is providing a directory for your test to store temporary files, which can easily be accomplished using the
Files.createTempDirectory()
API which works on both physical devices and under Ravenwood:
- One common use-case is providing a directory for your test to store temporary files, which can easily be accomplished using the
import java.nio.file.Files;
@RunWith(AndroidJUnit4.class)
public class MyTest {
@Before
public void setUp() throws Exception {
File tempDir = Files.createTempDirectory("MyTest").toFile();
...
- Although mocking code that your team doesn’t own is a generally discouraged testing practice, it can be a valuable pressure relief valve when a dependency isn’t yet supported.
Strategies for debugging test development
When writing tests you may encounter odd or hard to debug behaviors. One good place to start is at the beginning of the logs stored by atest:
$ atest MyTestsRavenwood
...
Test Logs have saved in /tmp/atest_result/20231128_094010_0e90t8v8/log
Run 'atest --history' to review test result history.
The most useful logs are in the isolated-java-logs
text file, which can typically be tab-completed by copy-pasting the logs path mentioned in the atest output:
$ less /tmp/atest_result/20231128_133105_h9al__79/log/i*/i*/isolated-java-logs*
Here are some common known issues and recommended workarounds:
- Some code may unconditionally interact with unsupported APIs, such as via static initializers. One strategy is to shift the logic into
@Before
methods and make it conditional by testingRavenwoodRule.isUnderRavenwood()
. - Some code may reference API symbols not yet present in the Ravenwood runtime, such as ART or ICU internals, or APIs from Mainline modules. One strategy is to refactor to avoid these internal dependencies, but Ravenwood aims to better support them soon.
- This may also manifest as very odd behavior, such as test not being executed at all, tracked by bug #312517322
- This may also manifest as an obscure Mockito error claiming “Mockito can only mock non-private & non-final classes”