Skip to main content
Version: 0.9.1

Filtering

Test filtering in marathon is done using allowlist and blocklist pattern.

tip

Test filtering works after a list of tests expected to run has been determined using test class regular expression. If your tests are not showing up after applying filtering logic - double-check the above parameter is not filtering it out.

Filtering logic

First allowlist is applied, then the blocklist. Each accepts a collection of TestFilter declarations and their parameters.

All the filters can be used in allowlist and in blocklist block as well, for example the following will run only smoke tests:

filteringConfiguration:
allowlist:
- type: "annotation"
values:
- "com.example.SmokeTest"

And the next snippet will execute everything, but the smoke tests:

filteringConfiguration:
blocklist:
- type: "annotation"
values:
- "com.example.SmokeTest"

"fully-qualified-test-name"

Filters tests by their FQTN which is $package.$class#$method. The # sign is important!

"fully-qualified-class-name"

Filters tests by their FQCN which is $package.$class |

"simple-class-name"

Filters tests by using only test class name, e.g. MyTest

"package"

Filters tests by using only test package, e.g. com.example

"method"

Filters tests by using only test method, e.g. myComplicatedTest

"annotation"

Filters tests by using only test annotation name, e.g. androidx.test.filters.LargeTest

"allure"

Filters tests by using allure-test-filter, source

Gradle plugin mapping

YAML typeGradle class
"fully-qualified-test-name"FullyQualifiedTestnameFilterConfiguration
"fully-qualified-class-name"FullyQualifiedClassnameFilterConfiguration
"simple-class-name"SimpleClassnameFilterConfiguration
"package"TestPackageFilterConfiguration
"method"TestMethodFilterConfiguration
"annotation"AnnotationFilterConfiguration
"allure"AllureFilterConfiguration
caution

Gradle will require you to import the filtering classes just as any Groovy/Kotlin code would. Using Kotlin DSL will make it a bit simpler though.

Filter parameters

Each of the above filters expects only one of the following parameters:

  • A regex for matching
  • An array of values
  • A file that contains each value on a separate line (empty lines will be ignored)

Regex filtering

An example of regex filtering is executing any test for a particular package, e.g. for package: com.example and it's subpackages:

filteringConfiguration:
allowlist:
- type: "package"
regex: "com\.example.*"

Values filtering

You could also specify each package separately via values:

filteringConfiguration:
allowlist:
- type: "package"
values:
- "com.example"
- "com.example.subpackage"

Values file filtering

Or you can supply these packages via a file (be careful with the relative paths: they will be relative to the workdir of the process):

filteringConfiguration:
allowlist:
- type: "package"
file: "testing/myfilterfile"

Inside the testing/myfilterfile you should supply the values, each on a separate line:

com.example
com.example.subpackage

Groovy filtering

With almost every filtering configuration (except for annotationDataFilter) it is possible to have regex and values.

Providing one value will be mapped to regex in the generated Marathonfile, more than one will end up in values

marathon {
filteringConfiguration {
allowlist {
annotationFilter = ['com.example.SmokeTest']//mapped to regex
testMethodFilter = ['testMethod', 'testSomethingElse']//mapped to values
}
}
}

Common examples

Running only specific tests

A common scenario is to execute a list of tests. You can do this via the FQTN filter:

filteringConfiguration:
allowlist:
- type: "fully-qualified-test-name"
values:
- "com.example.ElaborateTest#testMethod"
- "com.example.subpackage.OtherTest#testSomethingElse"

Running only specific test classes

If you want to execute tests ScaryTest and FlakyTest for any package using the class name filter:

filteringConfiguration:
allowlist:
- type: "simple-class-name"
values:
- "ScaryTest"
- "FlakyTest"

Extracting filtering values into a separate file

In case you want to separate the filtering configuration from the Marathonfile you can supply a reference to an external file:

filteringConfiguration:
allowlist:
- type: "simple-class-name"
file: testing/myfilterfile
tip

This extraction approach works for any test filter that supports values.

Inside the testing/myfilterfile you should supply the same values, each on a separate line, e.g. fully qualified class name filter:

com.example.ScaryTest
com.example.subpackage.FlakyTest

Allure platform test filter

filteringConfiguration:
allowlist:
- type: "allure"

Composition filtering

Marathon supports filtering using multiple test filters at the same time using a composition filter. It accepts a list of base filters and also an operation such as UNION, INTERSECTION or SUBTRACT. You can create complex filters such as get all the tests starting with E2E but get only methods from there ending with Test. Composition filter is not supported by groovy gradle scripts, but is supported if you use gradle kts.

An important thing to mention is that by default platform specific ignore options are not taken into account. This is because a cross-platform test runner cannot account for all the possible test frameworks out there. However, each framework's ignore option can still be "explained" to marathon, e.g. JUnit's org.junit.Ignore annotation can be specified in the filtering configuration.

filteringConfiguration:
allowlist:
- type: "simple-class-name"
regex: ".*"
- type: "fully-qualified-class-name"
values:
- "com.example.MyTest"
- "com.example.MyOtherTest"
- type: "fully-qualified-class-name"
file: "testing/mytestfilter"
- type: "method"
regex: "."
- type: "composition"
filters:
- type: "package"
regex: ".*"
- type: "method"
regex: ".*"
op: "UNION"
blocklist:
- type: "package"
regex: ".*"
- type: "annotation"
regex: ".*"

Fragmented execution of tests

This is a test filter similar to sharded test execution that AOSP provides.

It is intended to be used in situations where it is not possible to connect multiple execution devices to a single test run, e.g. CI setup that can schedule parallel jobs each containing a single execution device. There are two parameters for using fragmentation:

  • count - the number of overall fragments (e.g. 10 parallel execution)
  • index - current execution index (in our case of 10 executions valid indexes are 0..9)

This is a dynamic programming technique, hence the results will be sub-optimal compared to connecting multiple devices to the same test run

filteringConfiguration:
allowlist:
- type: "fragmentation"
index: 0
count: 10

If you want to dynamically pass the index of the test run you can use yaml envvar interpolation, e.g.:

filteringConfiguration:
allowlist:
- type: "fragmentation"
index: ${MARATHON_FRAGMENT_INDEX}
count: 10

and then execute the testing as following:

$ MARATHON_FRAGMENT_INDEX=0 marathon

To pass the fragment index in gradle refer to the Gradle's dynamic project properties