| // Copyright 2014 The Bazel Authors. All rights reserved. |
| // |
| // 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.google.devtools.build.lib.testutil; |
| |
| import com.google.common.base.Predicate; |
| import com.google.common.base.Predicates; |
| import com.google.common.collect.Iterables; |
| import com.google.common.collect.Sets; |
| import com.google.devtools.build.lib.util.Classpath; |
| import com.google.devtools.build.lib.util.Classpath.ClassPathException; |
| import com.google.errorprone.annotations.CanIgnoreReturnValue; |
| import java.lang.reflect.Modifier; |
| import java.util.Comparator; |
| import java.util.LinkedHashSet; |
| import java.util.Set; |
| import junit.framework.TestCase; |
| import org.junit.runner.RunWith; |
| |
| /** |
| * A collector for test classes, for both JUnit 3 and 4. To be used in combination with {@link |
| * CustomSuite}. |
| */ |
| public final class TestSuiteBuilder { |
| |
| private Set<Class<?>> testClasses = Sets.newTreeSet(new TestClassNameComparator()); |
| private Predicate<Class<?>> matchClassPredicate = Predicates.alwaysTrue(); |
| |
| /** |
| * Adds the tests found (directly) in class {@code c} to the set of tests this builder will |
| * search. |
| */ |
| @CanIgnoreReturnValue |
| public TestSuiteBuilder addTestClass(Class<?> c) { |
| testClasses.add(c); |
| return this; |
| } |
| |
| /** |
| * Adds all the test classes (top-level or nested) found in package {@code pkgName} or its |
| * subpackages to the set of tests this builder will search. |
| */ |
| @CanIgnoreReturnValue |
| public TestSuiteBuilder addPackageRecursive(String pkgName) { |
| for (Class<?> c : getClassesRecursive(pkgName)) { |
| addTestClass(c); |
| } |
| return this; |
| } |
| |
| private Set<Class<?>> getClassesRecursive(String pkgName) { |
| Set<Class<?>> result = new LinkedHashSet<>(); |
| try { |
| for (Class<?> clazz : Classpath.findClasses(pkgName)) { |
| if (isTestClass(clazz)) { |
| result.add(clazz); |
| } |
| } |
| } catch (ClassPathException e) { |
| throw new AssertionError("Cannot retrieve classes: " + e.getMessage()); |
| } |
| return result; |
| } |
| |
| /** Specifies a predicate returns false for classes we want to exclude. */ |
| @CanIgnoreReturnValue |
| public TestSuiteBuilder matchClasses(Predicate<Class<?>> predicate) { |
| matchClassPredicate = predicate; |
| return this; |
| } |
| |
| /** |
| * Creates and returns a TestSuite containing the tests from the given |
| * classes and/or packages which matched the given tags. |
| */ |
| public Set<Class<?>> create() { |
| Set<Class<?>> result = new LinkedHashSet<>(); |
| for (Class<?> testClass : Iterables.filter(testClasses, matchClassPredicate)) { |
| result.add(testClass); |
| } |
| return result; |
| } |
| |
| /** |
| * Determines if a given class is a test class. |
| * |
| * @param container class to test |
| * @return <code>true</code> if the test is a test class. |
| */ |
| private static boolean isTestClass(Class<?> container) { |
| return (isJunit4Test(container) || isJunit3Test(container)) |
| && !isSuite(container) |
| && !Modifier.isAbstract(container.getModifiers()); |
| } |
| |
| private static boolean isJunit4Test(Class<?> container) { |
| return container.isAnnotationPresent(RunWith.class); |
| } |
| |
| private static boolean isJunit3Test(Class<?> container) { |
| return TestCase.class.isAssignableFrom(container); |
| } |
| |
| /** |
| * Classes that have a {@code RunWith} annotation for {@link ClasspathSuite} or {@link |
| * CustomSuite} are automatically excluded to avoid picking up the suite class itself. |
| */ |
| private static boolean isSuite(Class<?> container) { |
| RunWith runWith = container.getAnnotation(RunWith.class); |
| return (runWith != null) |
| && ((runWith.value() == ClasspathSuite.class) || (runWith.value() == CustomSuite.class)); |
| } |
| |
| private static class TestClassNameComparator implements Comparator<Class<?>> { |
| @Override |
| public int compare(Class<?> o1, Class<?> o2) { |
| return o1.getName().compareTo(o2.getName()); |
| } |
| } |
| } |