| /* |
| * Copyright (C) 2016 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package com.android.cts.core.runner; |
| |
| import android.os.Bundle; |
| |
| import com.android.cts.core.runner.filter.CoreExpectationFilter; |
| import com.android.cts.core.runner.filter.CoreTestModeFilter; |
| |
| import org.junit.runner.Description; |
| import org.junit.runner.manipulation.Filter; |
| import org.junit.runners.ParentRunner; |
| import org.junit.runners.Suite; |
| |
| import java.util.List; |
| import java.util.function.Predicate; |
| |
| /** |
| * Filter out tests/classes that are not requested or which are expected to fail. |
| * |
| * <p>This filter has to handle both a hierarchy of {@code Description descriptions} that looks |
| * something like this: |
| * <pre> |
| * Suite |
| * Suite |
| * Suite |
| * ParentRunner |
| * Test |
| * ... |
| * ... |
| * ParentRunner |
| * Test |
| * ... |
| * ... |
| * Suite |
| * ParentRunner |
| * Test |
| * ... |
| * ... |
| * ... |
| * </pre> |
| * |
| * <p>It cannot filter out the non-leaf nodes in the hierarchy, i.e. {@link Suite} and |
| * {@link ParentRunner}, as that would prevent it from traversing the hierarchy and finding |
| * the leaf nodes. |
| */ |
| public class ExpectationBasedFilter extends Filter { |
| |
| private final Predicate<Description> mFilter; |
| |
| /** |
| * Invoked and created by JUnit when the command option |
| * {@code -e filter com.android.cts.core.runner.ExpectationBasedFilter} is passed to |
| * the {@code adb shell am instrument} command. |
| */ |
| public ExpectationBasedFilter(Bundle args) { |
| mFilter = CoreExpectationFilter.createInstance(args) |
| .and(CoreTestModeFilter.createInstance(args)); |
| } |
| |
| @Override |
| public boolean shouldRun(Description description) { |
| // Only filter leaf nodes. The description is for a test if and only if it is a leaf node. |
| // Non-leaf nodes must not be filtered out as that would prevent leaf nodes from being |
| // visited in the case when we are traversing the hierarchy of classes. |
| Description testDescription = getTestDescription(description); |
| if (testDescription == null) { |
| return true; |
| } |
| return mFilter.test(testDescription); |
| } |
| |
| private Description getTestDescription(Description description) { |
| List<Description> children = description.getChildren(); |
| // An empty description is by definition a test. |
| if (children.isEmpty()) { |
| return description; |
| } |
| |
| // Handle initialization errors that were wrapped in an ErrorReportingRunner as a special |
| // case. This is needed because ErrorReportingRunner is treated as a suite of Throwables, |
| // (where each Throwable corresponds to a test called initializationError) and so its |
| // description contains children, one for each Throwable, and so is not treated as a test |
| // to filter. Unfortunately, it does not support Filterable so this filter is never applied |
| // to its children. |
| // See https://github.com/junit-team/junit/issues/1253 |
| Description child = children.get(0); |
| String methodName = child.getMethodName(); |
| if ("initializationError".equals(methodName)) { |
| return child; |
| } |
| |
| return null; |
| } |
| |
| @Override |
| public String describe() { |
| return "TestFilter"; |
| } |
| } |