/* * Copyright (c) 2014, 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 8054987 * @summary Test sharing of annotations between Executable/Field instances. * Sharing should not be noticeable when performing mutating * operations. * @run testng AnnotationSharing */ import java.lang.annotation.*; import java.lang.reflect.*; import org.testng.annotations.Test; public class AnnotationSharing { public static void main(String ... args) throws Exception { } @Test public void testMethodSharing() throws Exception { Method[] m1 = AnnotationSharing.class.getMethods(); Method[] m2 = AnnotationSharing.class.getMethods(); validateSharingSafelyObservable(m1, m2); } @Test public void testDeclaredMethodSharing() throws Exception { Method[] m3 = AnnotationSharing.class.getDeclaredMethods(); Method[] m4 = AnnotationSharing.class.getDeclaredMethods(); validateSharingSafelyObservable(m3, m4); } @Test public void testFieldSharing() throws Exception { Field[] f1 = AnnotationSharing.class.getFields(); Field[] f2 = AnnotationSharing.class.getFields(); validateSharingSafelyObservable(f1, f2); } @Test public void testDeclaredFieldsSharing() throws Exception { Field[] f3 = AnnotationSharing.class.getDeclaredFields(); Field[] f4 = AnnotationSharing.class.getDeclaredFields(); validateSharingSafelyObservable(f3, f4); } @Test public void testMethodSharingOccurs() throws Exception { Method mm1 = AnnotationSharing.class.getDeclaredMethod("m", (Class[])null); Method mm2 = AnnotationSharing.class.getDeclaredMethod("m", (Class[])null); validateAnnotationSharing(mm1, mm2); } @Test public void testMethodSharingIsSafe() throws Exception { Method mm1 = AnnotationSharing.class.getDeclaredMethod("m", (Class[])null); Method mm2 = AnnotationSharing.class.getDeclaredMethod("m", (Class[])null); validateAnnotationSharingIsSafe(mm1, mm2); validateArrayValues(mm1.getAnnotation(Baz.class), mm2.getAnnotation(Baz.class)); } @Test public void testFieldSharingOccurs() throws Exception { Field ff1 = AnnotationSharing.class.getDeclaredField("f"); Field ff2 = AnnotationSharing.class.getDeclaredField("f"); validateAnnotationSharing(ff1, ff2); } @Test public void testFieldSharingIsSafe() throws Exception { Field ff1 = AnnotationSharing.class.getDeclaredField("f"); Field ff2 = AnnotationSharing.class.getDeclaredField("f"); validateAnnotationSharingIsSafe(ff1, ff2); validateArrayValues(ff1.getAnnotation(Baz.class), ff2.getAnnotation(Baz.class)); } // Validate that AccessibleObject instances are not shared private static void validateSharingSafelyObservable(AccessibleObject[] m1, AccessibleObject[] m2) throws Exception { // Validate that setAccessible works for (AccessibleObject m : m1) m.setAccessible(false); for (AccessibleObject m : m2) m.setAccessible(true); for (AccessibleObject m : m1) if (m.isAccessible()) throw new RuntimeException(m + " should not be accessible"); for (AccessibleObject m : m2) if (!m.isAccessible()) throw new RuntimeException(m + " should be accessible"); // Validate that methods are still equal() for (int i = 0; i < m1.length; i++) if (!m1[i].equals(m2[i])) throw new RuntimeException(m1[i] + " and " + m2[i] + " should be equal()"); // Validate that the arrays aren't shared for (int i = 0; i < m1.length; i++) m1[i] = null; for (int i = 0; i < m2.length; i++) if (m2[i] == null) throw new RuntimeException("Detected sharing of AccessibleObject arrays"); } // Validate that annotations are shared private static void validateAnnotationSharing(AccessibleObject m1, AccessibleObject m2) { Bar b1 = m1.getAnnotation(Bar.class); Bar b2 = m2.getAnnotation(Bar.class); if (b1 != b2) throw new RuntimeException(b1 + " and " + b2 + " should be =="); } // Validate that Method instances representing the annotation elements // behave as intended private static void validateAnnotationSharingIsSafe(AccessibleObject m1, AccessibleObject m2) throws Exception { Bar b1 = m1.getAnnotation(Bar.class); Bar b2 = m2.getAnnotation(Bar.class); Method mm1 = b1.annotationType().getMethod("value", (Class[]) null); Method mm2 = b2.annotationType().getMethod("value", (Class[]) null); inner(mm1, mm2); mm1 = b1.getClass().getMethod("value", (Class[]) null); mm2 = b2.getClass().getMethod("value", (Class[]) null); inner(mm1, mm2); } private static void inner(Method mm1, Method mm2) throws Exception { if (!mm1.equals(mm2)) throw new RuntimeException(mm1 + " and " + mm2 + " should be equal()"); mm1.setAccessible(false); mm2.setAccessible(true); if (mm1.isAccessible()) throw new RuntimeException(mm1 + " should not be accessible"); if (!mm2.isAccessible()) throw new RuntimeException(mm2 + " should be accessible"); } // Validate that array element values are not shared private static void validateArrayValues(Baz a, Baz b) { String[] s1 = a.value(); String[] s2 = b.value(); s1[0] = "22"; if (!s2[0].equals("1")) throw new RuntimeException("Mutation of array elements should not be detectable"); } @Foo @Bar("val") @Baz({"1", "2"}) public void m() { return ; } @Foo @Bar("someValue") @Baz({"1", "22", "33"}) public Object f = new Object(); } @Retention(RetentionPolicy.RUNTIME) @interface Foo {} @Retention(RetentionPolicy.RUNTIME) @interface Bar { String value(); } @Retention(RetentionPolicy.RUNTIME) @interface Baz { String [] value(); }