AnnotationUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3;

  18. import java.lang.annotation.Annotation;
  19. import java.lang.reflect.Method;
  20. import java.util.Arrays;

  21. import org.apache.commons.lang3.builder.ToStringBuilder;
  22. import org.apache.commons.lang3.builder.ToStringStyle;
  23. import org.apache.commons.lang3.exception.UncheckedException;

  24. /**
  25.  * Helper methods for working with {@link Annotation} instances.
  26.  *
  27.  * <p>This class contains various utility methods that make working with
  28.  * annotations simpler.</p>
  29.  *
  30.  * <p>{@link Annotation} instances are always proxy objects; unfortunately
  31.  * dynamic proxies cannot be depended upon to know how to implement certain
  32.  * methods in the same manner as would be done by "natural" {@link Annotation}s.
  33.  * The methods presented in this class can be used to avoid that possibility. It
  34.  * is of course also possible for dynamic proxies to actually delegate their
  35.  * e.g. {@link Annotation#equals(Object)}/{@link Annotation#hashCode()}/
  36.  * {@link Annotation#toString()} implementations to {@link AnnotationUtils}.</p>
  37.  *
  38.  * <p>#ThreadSafe#</p>
  39.  *
  40.  * @since 3.0
  41.  */
  42. public class AnnotationUtils {

  43.     /**
  44.      * A style that prints annotations as recommended.
  45.      */
  46.     private static final ToStringStyle TO_STRING_STYLE = new ToStringStyle() {
  47.         /** Serialization version */
  48.         private static final long serialVersionUID = 1L;

  49.         {
  50.             setDefaultFullDetail(true);
  51.             setArrayContentDetail(true);
  52.             setUseClassName(true);
  53.             setUseShortClassName(true);
  54.             setUseIdentityHashCode(false);
  55.             setContentStart("(");
  56.             setContentEnd(")");
  57.             setFieldSeparator(", ");
  58.             setArrayStart("[");
  59.             setArrayEnd("]");
  60.         }

  61.         /**
  62.          * {@inheritDoc}
  63.          */
  64.         @Override
  65.         protected void appendDetail(final StringBuffer buffer, final String fieldName, Object value) {
  66.             if (value instanceof Annotation) {
  67.                 value = AnnotationUtils.toString((Annotation) value);
  68.             }
  69.             super.appendDetail(buffer, fieldName, value);
  70.         }

  71.         /**
  72.          * {@inheritDoc}
  73.          */
  74.         @Override
  75.         protected String getShortClassName(final Class<?> cls) {
  76.             // formatter:off
  77.             return ClassUtils.getAllInterfaces(cls).stream().filter(Annotation.class::isAssignableFrom).findFirst()
  78.                 .map(iface -> "@" + iface.getName())
  79.                 .orElse(StringUtils.EMPTY);
  80.             // formatter:on
  81.         }

  82.     };

  83.     /**
  84.      * Helper method for comparing two arrays of annotations.
  85.      *
  86.      * @param a1 the first array
  87.      * @param a2 the second array
  88.      * @return a flag whether these arrays are equal
  89.      */
  90.     private static boolean annotationArrayMemberEquals(final Annotation[] a1, final Annotation[] a2) {
  91.         if (a1.length != a2.length) {
  92.             return false;
  93.         }
  94.         for (int i = 0; i < a1.length; i++) {
  95.             if (!equals(a1[i], a2[i])) {
  96.                 return false;
  97.             }
  98.         }
  99.         return true;
  100.     }

  101.     /**
  102.      * Helper method for comparing two objects of an array type.
  103.      *
  104.      * @param componentType the component type of the array
  105.      * @param o1 the first object
  106.      * @param o2 the second object
  107.      * @return a flag whether these objects are equal
  108.      */
  109.     private static boolean arrayMemberEquals(final Class<?> componentType, final Object o1, final Object o2) {
  110.         if (componentType.isAnnotation()) {
  111.             return annotationArrayMemberEquals((Annotation[]) o1, (Annotation[]) o2);
  112.         }
  113.         if (componentType.equals(Byte.TYPE)) {
  114.             return Arrays.equals((byte[]) o1, (byte[]) o2);
  115.         }
  116.         if (componentType.equals(Short.TYPE)) {
  117.             return Arrays.equals((short[]) o1, (short[]) o2);
  118.         }
  119.         if (componentType.equals(Integer.TYPE)) {
  120.             return Arrays.equals((int[]) o1, (int[]) o2);
  121.         }
  122.         if (componentType.equals(Character.TYPE)) {
  123.             return Arrays.equals((char[]) o1, (char[]) o2);
  124.         }
  125.         if (componentType.equals(Long.TYPE)) {
  126.             return Arrays.equals((long[]) o1, (long[]) o2);
  127.         }
  128.         if (componentType.equals(Float.TYPE)) {
  129.             return Arrays.equals((float[]) o1, (float[]) o2);
  130.         }
  131.         if (componentType.equals(Double.TYPE)) {
  132.             return Arrays.equals((double[]) o1, (double[]) o2);
  133.         }
  134.         if (componentType.equals(Boolean.TYPE)) {
  135.             return Arrays.equals((boolean[]) o1, (boolean[]) o2);
  136.         }
  137.         return Arrays.equals((Object[]) o1, (Object[]) o2);
  138.     }

  139.     /**
  140.      * Helper method for generating a hash code for an array.
  141.      *
  142.      * @param componentType the component type of the array
  143.      * @param o the array
  144.      * @return a hash code for the specified array
  145.      */
  146.     private static int arrayMemberHash(final Class<?> componentType, final Object o) {
  147.         if (componentType.equals(Byte.TYPE)) {
  148.             return Arrays.hashCode((byte[]) o);
  149.         }
  150.         if (componentType.equals(Short.TYPE)) {
  151.             return Arrays.hashCode((short[]) o);
  152.         }
  153.         if (componentType.equals(Integer.TYPE)) {
  154.             return Arrays.hashCode((int[]) o);
  155.         }
  156.         if (componentType.equals(Character.TYPE)) {
  157.             return Arrays.hashCode((char[]) o);
  158.         }
  159.         if (componentType.equals(Long.TYPE)) {
  160.             return Arrays.hashCode((long[]) o);
  161.         }
  162.         if (componentType.equals(Float.TYPE)) {
  163.             return Arrays.hashCode((float[]) o);
  164.         }
  165.         if (componentType.equals(Double.TYPE)) {
  166.             return Arrays.hashCode((double[]) o);
  167.         }
  168.         if (componentType.equals(Boolean.TYPE)) {
  169.             return Arrays.hashCode((boolean[]) o);
  170.         }
  171.         return Arrays.hashCode((Object[]) o);
  172.     }

  173.     /**
  174.      * Checks if two annotations are equal using the criteria for equality
  175.      * presented in the {@link Annotation#equals(Object)} API docs.
  176.      *
  177.      * @param a1 the first Annotation to compare, {@code null} returns
  178.      * {@code false} unless both are {@code null}
  179.      * @param a2 the second Annotation to compare, {@code null} returns
  180.      * {@code false} unless both are {@code null}
  181.      * @return {@code true} if the two annotations are {@code equal} or both
  182.      * {@code null}
  183.      */
  184.     public static boolean equals(final Annotation a1, final Annotation a2) {
  185.         if (a1 == a2) {
  186.             return true;
  187.         }
  188.         if (a1 == null || a2 == null) {
  189.             return false;
  190.         }
  191.         final Class<? extends Annotation> type1 = a1.annotationType();
  192.         final Class<? extends Annotation> type2 = a2.annotationType();
  193.         Validate.notNull(type1, "Annotation %s with null annotationType()", a1);
  194.         Validate.notNull(type2, "Annotation %s with null annotationType()", a2);
  195.         if (!type1.equals(type2)) {
  196.             return false;
  197.         }
  198.         try {
  199.             for (final Method m : type1.getDeclaredMethods()) {
  200.                 if (m.getParameterTypes().length == 0
  201.                         && isValidAnnotationMemberType(m.getReturnType())) {
  202.                     final Object v1 = m.invoke(a1);
  203.                     final Object v2 = m.invoke(a2);
  204.                     if (!memberEquals(m.getReturnType(), v1, v2)) {
  205.                         return false;
  206.                     }
  207.                 }
  208.             }
  209.         } catch (final ReflectiveOperationException ex) {
  210.             return false;
  211.         }
  212.         return true;
  213.     }

  214.     /**
  215.      * Generate a hash code for the given annotation using the algorithm
  216.      * presented in the {@link Annotation#hashCode()} API docs.
  217.      *
  218.      * @param a the Annotation for a hash code calculation is desired, not
  219.      * {@code null}
  220.      * @return the calculated hash code
  221.      * @throws RuntimeException if an {@link Exception} is encountered during
  222.      * annotation member access
  223.      * @throws IllegalStateException if an annotation method invocation returns
  224.      * {@code null}
  225.      */
  226.     public static int hashCode(final Annotation a) {
  227.         int result = 0;
  228.         final Class<? extends Annotation> type = a.annotationType();
  229.         for (final Method m : type.getDeclaredMethods()) {
  230.             try {
  231.                 final Object value = m.invoke(a);
  232.                 if (value == null) {
  233.                     throw new IllegalStateException(String.format("Annotation method %s returned null", m));
  234.                 }
  235.                 result += hashMember(m.getName(), value);
  236.             } catch (final ReflectiveOperationException ex) {
  237.                 throw new UncheckedException(ex);
  238.             }
  239.         }
  240.         return result;
  241.     }

  242.     //besides modularity, this has the advantage of autoboxing primitives:
  243.     /**
  244.      * Helper method for generating a hash code for a member of an annotation.
  245.      *
  246.      * @param name the name of the member
  247.      * @param value the value of the member
  248.      * @return a hash code for this member
  249.      */
  250.     private static int hashMember(final String name, final Object value) {
  251.         final int part1 = name.hashCode() * 127;
  252.         if (ObjectUtils.isArray(value)) {
  253.             return part1 ^ arrayMemberHash(value.getClass().getComponentType(), value);
  254.         }
  255.         if (value instanceof Annotation) {
  256.             return part1 ^ hashCode((Annotation) value);
  257.         }
  258.         return part1 ^ value.hashCode();
  259.     }

  260.     /**
  261.      * Checks if the specified type is permitted as an annotation member.
  262.      *
  263.      * <p>The Java language specification only permits certain types to be used
  264.      * in annotations. These include {@link String}, {@link Class}, primitive
  265.      * types, {@link Annotation}, {@link Enum}, and single-dimensional arrays of
  266.      * these types.</p>
  267.      *
  268.      * @param type the type to check, {@code null}
  269.      * @return {@code true} if the type is a valid type to use in an annotation
  270.      */
  271.     public static boolean isValidAnnotationMemberType(Class<?> type) {
  272.         if (type == null) {
  273.             return false;
  274.         }
  275.         if (type.isArray()) {
  276.             type = type.getComponentType();
  277.         }
  278.         return type.isPrimitive() || type.isEnum() || type.isAnnotation()
  279.                 || String.class.equals(type) || Class.class.equals(type);
  280.     }

  281.     /**
  282.      * Helper method for checking whether two objects of the given type are
  283.      * equal. This method is used to compare the parameters of two annotation
  284.      * instances.
  285.      *
  286.      * @param type the type of the objects to be compared
  287.      * @param o1 the first object
  288.      * @param o2 the second object
  289.      * @return a flag whether these objects are equal
  290.      */
  291.     private static boolean memberEquals(final Class<?> type, final Object o1, final Object o2) {
  292.         if (o1 == o2) {
  293.             return true;
  294.         }
  295.         if (o1 == null || o2 == null) {
  296.             return false;
  297.         }
  298.         if (type.isArray()) {
  299.             return arrayMemberEquals(type.getComponentType(), o1, o2);
  300.         }
  301.         if (type.isAnnotation()) {
  302.             return equals((Annotation) o1, (Annotation) o2);
  303.         }
  304.         return o1.equals(o2);
  305.     }

  306.     /**
  307.      * Generate a string representation of an Annotation, as suggested by
  308.      * {@link Annotation#toString()}.
  309.      *
  310.      * @param a the annotation of which a string representation is desired
  311.      * @return the standard string representation of an annotation, not
  312.      * {@code null}
  313.      */
  314.     public static String toString(final Annotation a) {
  315.         final ToStringBuilder builder = new ToStringBuilder(a, TO_STRING_STYLE);
  316.         for (final Method m : a.annotationType().getDeclaredMethods()) {
  317.             if (m.getParameterTypes().length > 0) {
  318.                 continue; // wtf?
  319.             }
  320.             try {
  321.                 builder.append(m.getName(), m.invoke(a));
  322.             } catch (final ReflectiveOperationException ex) {
  323.                 throw new UncheckedException(ex);
  324.             }
  325.         }
  326.         return builder.build();
  327.     }

  328.     /**
  329.      * {@link AnnotationUtils} instances should NOT be constructed in
  330.      * standard programming. Instead, the class should be used statically.
  331.      *
  332.      * <p>This constructor is public to permit tools that require a JavaBean
  333.      * instance to operate.</p>
  334.      * @deprecated TODO Make private in 4.0.
  335.      */
  336.     @Deprecated
  337.     public AnnotationUtils() {
  338.         // empty
  339.     }
  340. }