/* * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* * @test * @bug 8011940 * @summary Test inheritance, order and class redefinition behaviour of RUNTIME * class annotations * @author plevart * @modules java.base/java.lang:open * java.base/sun.reflect.annotation */ import sun.reflect.annotation.AnnotationParser; import java.lang.annotation.Annotation; import java.lang.annotation.Inherited; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.List; import java.util.StringJoiner; public class AnnotationsInheritanceOrderRedefinitionTest { @Retention(RetentionPolicy.RUNTIME) @Inherited @interface Ann1 { String value(); } @Retention(RetentionPolicy.RUNTIME) @Inherited @interface Ann2 { String value(); } @Retention(RetentionPolicy.RUNTIME) @Inherited @interface Ann3 { String value(); } @Ann1("A") @Ann2("A") static class A {} @Ann3("B") static class B extends A {} @Ann1("C") @Ann3("C") static class C extends B {} public static void main(String[] args) { StringBuilder msgs = new StringBuilder(); boolean ok = true; ok &= annotationsEqual(msgs, A.class, true, ann(Ann1.class, "A"), ann(Ann2.class, "A")); ok &= annotationsEqual(msgs, A.class, false, ann(Ann1.class, "A"), ann(Ann2.class, "A")); ok &= annotationsEqual(msgs, B.class, true, ann(Ann3.class, "B")); ok &= annotationsEqual(msgs, B.class, false, ann(Ann1.class, "A"), ann(Ann2.class, "A"), ann(Ann3.class, "B")); ok &= annotationsEqual(msgs, C.class, true, ann(Ann1.class, "C"), ann(Ann3.class, "C")); ok &= annotationsEqual(msgs, C.class, false, ann(Ann1.class, "C"), ann(Ann2.class, "A"), ann(Ann3.class, "C")); Annotation[] declaredAnnotatiosA = A.class.getDeclaredAnnotations(); Annotation[] annotationsA = A.class.getAnnotations(); Annotation[] declaredAnnotatiosB = B.class.getDeclaredAnnotations(); Annotation[] annotationsB = B.class.getAnnotations(); Annotation[] declaredAnnotatiosC = C.class.getDeclaredAnnotations(); Annotation[] annotationsC = C.class.getAnnotations(); incrementClassRedefinedCount(A.class); incrementClassRedefinedCount(B.class); incrementClassRedefinedCount(C.class); ok &= annotationsEqualButNotSame(msgs, A.class, true, declaredAnnotatiosA); ok &= annotationsEqualButNotSame(msgs, A.class, false, annotationsA); ok &= annotationsEqualButNotSame(msgs, B.class, true, declaredAnnotatiosB); ok &= annotationsEqualButNotSame(msgs, B.class, false, annotationsB); ok &= annotationsEqualButNotSame(msgs, C.class, true, declaredAnnotatiosC); ok &= annotationsEqualButNotSame(msgs, C.class, false, annotationsC); if (!ok) { throw new RuntimeException("test failure\n" + msgs); } } // utility methods private static boolean annotationsEqualButNotSame(StringBuilder msgs, Class declaringClass, boolean declaredOnly, Annotation[] oldAnns) { if (!annotationsEqual(msgs, declaringClass, declaredOnly, oldAnns)) { return false; } Annotation[] anns = declaredOnly ? declaringClass.getDeclaredAnnotations() : declaringClass.getAnnotations(); List sameAnns = new ArrayList<>(); for (int i = 0; i < anns.length; i++) { if (anns[i] == oldAnns[i]) { sameAnns.add(anns[i]); } } if (!sameAnns.isEmpty()) { msgs.append(declaredOnly ? "declared " : "").append("annotations for ") .append(declaringClass.getSimpleName()) .append(" not re-parsed after class redefinition: ") .append(toSimpleString(sameAnns)).append("\n"); return false; } else { return true; } } private static boolean annotationsEqual(StringBuilder msgs, Class declaringClass, boolean declaredOnly, Annotation... expectedAnns) { Annotation[] anns = declaredOnly ? declaringClass.getDeclaredAnnotations() : declaringClass.getAnnotations(); if (!Arrays.equals(anns, expectedAnns)) { msgs.append(declaredOnly ? "declared " : "").append("annotations for ") .append(declaringClass.getSimpleName()).append(" are: ") .append(toSimpleString(anns)).append(", expected: ") .append(toSimpleString(expectedAnns)).append("\n"); return false; } else { return true; } } private static Annotation ann(Class annotationType, Object value) { return AnnotationParser.annotationForMap(annotationType, Collections.singletonMap("value", value)); } private static String toSimpleString(List anns) { return toSimpleString(anns.toArray(new Annotation[anns.size()])); } private static String toSimpleString(Annotation[] anns) { StringJoiner joiner = new StringJoiner(", "); for (Annotation ann : anns) { joiner.add(toSimpleString(ann)); } return joiner.toString(); } private static String toSimpleString(Annotation ann) { Class annotationType = ann.annotationType(); Object value; try { value = annotationType.getDeclaredMethod("value").invoke(ann); } catch (IllegalAccessException | InvocationTargetException | NoSuchMethodException e) { throw new RuntimeException(e); } return "@" + annotationType.getSimpleName() + "(" + value + ")"; } private static final Field classRedefinedCountField; static { try { classRedefinedCountField = Class.class.getDeclaredField("classRedefinedCount"); classRedefinedCountField.setAccessible(true); } catch (NoSuchFieldException e) { throw new Error(e); } } private static void incrementClassRedefinedCount(Class clazz) { try { classRedefinedCountField.set(clazz, ((Integer) classRedefinedCountField.get(clazz)) + 1); } catch (IllegalAccessException e) { throw new RuntimeException(e); } } }