1 /*
2  * Copyright (C) 2008 The Guava Authors
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.google.common.collect.testing.features;
18 
19 import com.google.common.collect.testing.Helpers;
20 
21 import java.lang.annotation.Annotation;
22 import java.lang.reflect.AnnotatedElement;
23 import java.lang.reflect.Method;
24 import java.util.ArrayList;
25 import java.util.HashMap;
26 import java.util.LinkedHashSet;
27 import java.util.List;
28 import java.util.Map;
29 import java.util.Set;
30 
31 /**
32  * Utilities for collecting and validating tester requirements from annotations.
33  *
34  * @author George van den Driessche
35  */
36 public class FeatureUtil {
37   /**
38    * A cache of annotated objects (typically a Class or Method) to its
39    * set of annotations.
40    */
41   private static Map<AnnotatedElement, Annotation[]> annotationCache =
42       new HashMap<AnnotatedElement, Annotation[]>();
43 
44   private static final Map<Class<?>, TesterRequirements>
45       classTesterRequirementsCache =
46           new HashMap<Class<?>, TesterRequirements>();
47 
48   /**
49    * Given a set of features, add to it all the features directly or indirectly
50    * implied by any of them, and return it.
51    * @param features the set of features to expand
52    * @return the same set of features, expanded with all implied features
53    */
addImpliedFeatures(Set<Feature<?>> features)54   public static Set<Feature<?>> addImpliedFeatures(Set<Feature<?>> features) {
55     // The base case of the recursion is an empty set of features, which will
56     // occur when the previous set contained only simple features.
57     if (!features.isEmpty()) {
58       features.addAll(impliedFeatures(features));
59     }
60     return features;
61   }
62 
63   /**
64    * Given a set of features, return a new set of all features directly or
65    * indirectly implied by any of them.
66    * @param features the set of features whose implications to find
67    * @return the implied set of features
68    */
impliedFeatures(Set<Feature<?>> features)69   public static Set<Feature<?>> impliedFeatures(Set<Feature<?>> features) {
70     Set<Feature<?>> implied = new LinkedHashSet<Feature<?>>();
71     for (Feature<?> feature : features) {
72       implied.addAll(feature.getImpliedFeatures());
73     }
74     addImpliedFeatures(implied);
75     return implied;
76   }
77 
78   /**
79    * Get the full set of requirements for a tester class.
80    * @param testerClass a tester class
81    * @return all the constraints implicitly or explicitly required by the class
82    * or any of its superclasses.
83    * @throws ConflictingRequirementsException if the requirements are mutually
84    * inconsistent.
85    */
getTesterRequirements(Class<?> testerClass)86   public static TesterRequirements getTesterRequirements(Class<?> testerClass)
87       throws ConflictingRequirementsException {
88     synchronized (classTesterRequirementsCache) {
89       TesterRequirements requirements =
90           classTesterRequirementsCache.get(testerClass);
91       if (requirements == null) {
92         requirements = buildTesterRequirements(testerClass);
93         classTesterRequirementsCache.put(testerClass, requirements);
94       }
95       return requirements;
96     }
97   }
98 
99   /**
100    * Get the full set of requirements for a tester class.
101    * @param testerMethod a test method of a tester class
102    * @return all the constraints implicitly or explicitly required by the
103    * method, its declaring class, or any of its superclasses.
104    * @throws ConflictingRequirementsException if the requirements are
105    * mutually inconsistent.
106    */
getTesterRequirements(Method testerMethod)107   public static TesterRequirements getTesterRequirements(Method testerMethod)
108       throws ConflictingRequirementsException {
109     return buildTesterRequirements(testerMethod);
110   }
111 
112   /**
113    * Construct the full set of requirements for a tester class.
114    * @param testerClass a tester class
115    * @return all the constraints implicitly or explicitly required by the class
116    * or any of its superclasses.
117    * @throws ConflictingRequirementsException if the requirements are mutually
118    * inconsistent.
119    */
buildTesterRequirements(Class<?> testerClass)120   static TesterRequirements buildTesterRequirements(Class<?> testerClass)
121       throws ConflictingRequirementsException {
122     final TesterRequirements declaredRequirements =
123         buildDeclaredTesterRequirements(testerClass);
124     Class<?> baseClass = testerClass.getSuperclass();
125     if (baseClass == null) {
126       return declaredRequirements;
127     } else {
128       final TesterRequirements clonedBaseRequirements =
129           new TesterRequirements(getTesterRequirements(baseClass));
130       return incorporateRequirements(
131           clonedBaseRequirements, declaredRequirements, testerClass);
132     }
133   }
134 
135   /**
136    * Construct the full set of requirements for a tester method.
137    * @param testerMethod a test method of a tester class
138    * @return all the constraints implicitly or explicitly required by the
139    * method, its declaring class, or any of its superclasses.
140    * @throws ConflictingRequirementsException if the requirements are mutually
141    * inconsistent.
142    */
buildTesterRequirements(Method testerMethod)143   static TesterRequirements buildTesterRequirements(Method testerMethod)
144       throws ConflictingRequirementsException {
145     TesterRequirements clonedClassRequirements = new TesterRequirements(
146         getTesterRequirements(testerMethod.getDeclaringClass()));
147     TesterRequirements declaredRequirements =
148         buildDeclaredTesterRequirements(testerMethod);
149     return incorporateRequirements(
150         clonedClassRequirements, declaredRequirements, testerMethod);
151   }
152 
153   /**
154    * Construct the set of requirements specified by annotations
155    * directly on a tester class or method.
156    * @param classOrMethod a tester class or a test method thereof
157    * @return all the constraints implicitly or explicitly required by
158    *         annotations on the class or method.
159    * @throws ConflictingRequirementsException if the requirements are mutually
160    *         inconsistent.
161    */
buildDeclaredTesterRequirements( AnnotatedElement classOrMethod)162   public static TesterRequirements buildDeclaredTesterRequirements(
163       AnnotatedElement classOrMethod)
164       throws ConflictingRequirementsException {
165     TesterRequirements requirements = new TesterRequirements();
166 
167     Iterable<Annotation> testerAnnotations =
168         getTesterAnnotations(classOrMethod);
169     for (Annotation testerAnnotation : testerAnnotations) {
170       TesterRequirements moreRequirements =
171           buildTesterRequirements(testerAnnotation);
172       incorporateRequirements(
173           requirements, moreRequirements, testerAnnotation);
174     }
175 
176     return requirements;
177   }
178 
179   /**
180    * Find all the tester annotations declared on a tester class or method.
181    * @param classOrMethod a class or method whose tester annotations to find
182    * @return an iterable sequence of tester annotations on the class
183    */
getTesterAnnotations( AnnotatedElement classOrMethod)184   public static Iterable<Annotation> getTesterAnnotations(
185       AnnotatedElement classOrMethod) {
186     List<Annotation> result = new ArrayList<Annotation>();
187 
188     Annotation[] annotations;
189     synchronized (annotationCache) {
190       annotations = annotationCache.get(classOrMethod);
191       if (annotations == null) {
192         annotations = classOrMethod.getDeclaredAnnotations();
193         annotationCache.put(classOrMethod, annotations);
194       }
195     }
196 
197     for (Annotation a : annotations) {
198       Class<? extends Annotation> annotationClass = a.annotationType();
199       if (annotationClass.isAnnotationPresent(TesterAnnotation.class)) {
200         result.add(a);
201       }
202     }
203     return result;
204   }
205 
206   /**
207    * Find all the constraints explicitly or implicitly specified by a single
208    * tester annotation.
209    * @param testerAnnotation a tester annotation
210    * @return the requirements specified by the annotation
211    * @throws ConflictingRequirementsException if the requirements are mutually
212    *         inconsistent.
213    */
buildTesterRequirements( Annotation testerAnnotation)214   private static TesterRequirements buildTesterRequirements(
215       Annotation testerAnnotation)
216       throws ConflictingRequirementsException {
217     Class<? extends Annotation> annotationClass = testerAnnotation.getClass();
218     final Feature<?>[] presentFeatures;
219     final Feature<?>[] absentFeatures;
220     try {
221       presentFeatures = (Feature[]) annotationClass.getMethod("value")
222           .invoke(testerAnnotation);
223       absentFeatures = (Feature[]) annotationClass.getMethod("absent")
224           .invoke(testerAnnotation);
225     } catch (Exception e) {
226       throw new IllegalArgumentException(
227           "Error extracting features from tester annotation.", e);
228     }
229     Set<Feature<?>> allPresentFeatures =
230         addImpliedFeatures(Helpers.<Feature<?>>copyToSet(presentFeatures));
231     Set<Feature<?>> allAbsentFeatures =
232         addImpliedFeatures(Helpers.<Feature<?>>copyToSet(absentFeatures));
233     Set<Feature<?>> conflictingFeatures =
234         intersection(allPresentFeatures, allAbsentFeatures);
235     if (!conflictingFeatures.isEmpty()) {
236       throw new ConflictingRequirementsException("Annotation explicitly or " +
237           "implicitly requires one or more features to be both present " +
238           "and absent.",
239           conflictingFeatures, testerAnnotation);
240     }
241     return new TesterRequirements(allPresentFeatures, allAbsentFeatures);
242   }
243 
244   /**
245    * Incorporate additional requirements into an existing requirements object.
246    * @param requirements the existing requirements object
247    * @param moreRequirements more requirements to incorporate
248    * @param source the source of the additional requirements
249    *        (used only for error reporting)
250    * @return the existing requirements object, modified to include the
251    *         additional requirements
252    * @throws ConflictingRequirementsException if the additional requirements
253    *         are inconsistent with the existing requirements
254    */
incorporateRequirements( TesterRequirements requirements, TesterRequirements moreRequirements, Object source)255   private static TesterRequirements incorporateRequirements(
256       TesterRequirements requirements, TesterRequirements moreRequirements,
257       Object source) throws ConflictingRequirementsException {
258     Set<Feature<?>> presentFeatures = requirements.getPresentFeatures();
259     Set<Feature<?>> absentFeatures = requirements.getAbsentFeatures();
260     Set<Feature<?>> morePresentFeatures = moreRequirements.getPresentFeatures();
261     Set<Feature<?>> moreAbsentFeatures = moreRequirements.getAbsentFeatures();
262     checkConflict(
263         "absent", absentFeatures,
264         "present", morePresentFeatures, source);
265     checkConflict(
266         "present", presentFeatures,
267         "absent", moreAbsentFeatures, source);
268     presentFeatures.addAll(morePresentFeatures);
269     absentFeatures.addAll(moreAbsentFeatures);
270     return requirements;
271   }
272 
273   // Used by incorporateRequirements() only
checkConflict( String earlierRequirement, Set<Feature<?>> earlierFeatures, String newRequirement, Set<Feature<?>> newFeatures, Object source)274   private static void checkConflict(
275       String earlierRequirement, Set<Feature<?>> earlierFeatures,
276       String newRequirement, Set<Feature<?>> newFeatures,
277       Object source) throws ConflictingRequirementsException {
278     Set<Feature<?>> conflictingFeatures;
279     conflictingFeatures = intersection(newFeatures, earlierFeatures);
280     if (!conflictingFeatures.isEmpty()) {
281       throw new ConflictingRequirementsException(String.format(
282           "Annotation requires to be %s features that earlier " +
283           "annotations required to be %s.",
284               newRequirement, earlierRequirement),
285           conflictingFeatures, source);
286     }
287   }
288 
289   /**
290    * Construct a new {@link java.util.Set} that is the intersection
291    * of the given sets.
292    */
293   // Calls generic varargs method.
294   @SuppressWarnings("unchecked")
intersection( Set<? extends T> set1, Set<? extends T> set2)295   public static <T> Set<T> intersection(
296       Set<? extends T> set1, Set<? extends T> set2) {
297     return intersection(new Set[] {set1, set2});
298   }
299 
300   /**
301    * Construct a new {@link java.util.Set} that is the intersection
302    * of all the given sets.
303    * @param sets the sets to intersect
304    * @return the intersection of the sets
305    * @throws java.lang.IllegalArgumentException if there are no sets to
306    *         intersect
307    */
intersection(Set<? extends T> .... sets)308   public static <T> Set<T> intersection(Set<? extends T> ... sets) {
309     if (sets.length == 0) {
310       throw new IllegalArgumentException(
311           "Can't intersect no sets; would have to return the universe.");
312     }
313     Set<T> results = Helpers.copyToSet(sets[0]);
314     for (int i = 1; i < sets.length; i++) {
315       Set<? extends T> set = sets[i];
316       results.retainAll(set);
317     }
318     return results;
319   }
320 }
321