| /* |
| * Copyright (c) 2015, 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 8058595 |
| * @summary Test that AnnotatedType.getAnnotatedOwnerType() works as expected |
| * |
| * @library /lib/testlibrary |
| * @build jdk.testlibrary.Asserts |
| * @run main GetAnnotatedOwnerType |
| */ |
| |
| import java.lang.annotation.*; |
| import java.lang.reflect.*; |
| |
| import jdk.testlibrary.Asserts; |
| |
| public class GetAnnotatedOwnerType<Dummy> { |
| public @TA("generic") GetAnnotatedOwnerType<String> . @TB("generic") Nested<Integer> genericField; |
| public @TA("raw") GetAnnotatedOwnerType . @TB("raw") Nested rawField; |
| public @TA("non-generic") GetAnnotatedOwnerTypeAuxilliary . @TB("non-generic") Inner nonGeneric; |
| public @TA("non-generic") GetAnnotatedOwnerTypeAuxilliary . @TB("generic") InnerGeneric<String> innerGeneric; |
| public @TA("non-generic") GetAnnotatedOwnerTypeAuxilliary . @TB("raw") InnerGeneric innerRaw; |
| public Object anonymous = new Object() {}; |
| public @TA("array") Dummy[] dummy; |
| public @TA("wildcard") GetAnnotatedOwnerType<?> wildcard; |
| public @TA("typevariable") Dummy tv; |
| public @TA("bad") GetAnnotatedOwnerType<@TA("good") GetAnnotatedOwnerType<String> . @TB("tb") Nested<Integer> > typeArgument; |
| public GetAnnotatedOwnerType< GetAnnotatedOwnerType<String> . |
| B . |
| C<Class<?>, ? extends @TA("complicated") Exception> . |
| D<Number> > [] complicated; |
| |
| public static void main(String[] args) throws Exception { |
| testGeneric(); |
| testRaw(); |
| testNonGeneric(); |
| testInnerGeneric(); |
| testInnerRaw(); |
| |
| testLocalClass(); |
| testAnonymousClass(); |
| |
| testArray(); |
| testWildcard(); |
| testTypeParameter(); |
| |
| testTypeArgument(); |
| testComplicated(); |
| } |
| |
| public static void testGeneric() throws Exception { |
| Field f = GetAnnotatedOwnerType.class.getField("genericField"); |
| |
| // make sure inner is correctly annotated |
| AnnotatedType inner = f.getAnnotatedType(); |
| Asserts.assertEquals(inner.getAnnotation(TB.class).value(), "generic"); |
| Asserts.assertTrue(inner.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + inner.getAnnotations().length); |
| |
| // make sure owner is correctly annotated, on the correct type |
| AnnotatedType outer = inner.getAnnotatedOwnerType(); |
| Asserts.assertEquals(outer.getType(), ((ParameterizedType) f.getGenericType()).getOwnerType()); |
| Asserts.assertEquals(outer.getAnnotation(TA.class).value(), "generic"); |
| Asserts.assertTrue(outer.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + outer.getAnnotations().length); |
| } |
| |
| public static void testRaw() throws Exception { |
| Field f = GetAnnotatedOwnerType.class.getField("rawField"); |
| |
| // make sure inner is correctly annotated |
| AnnotatedType inner = f.getAnnotatedType(); |
| Asserts.assertEquals(inner.getAnnotation(TB.class).value(), "raw"); |
| Asserts.assertTrue(inner.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + inner.getAnnotations().length); |
| |
| // make sure owner is correctly annotated, on the correct type |
| AnnotatedType outer = inner.getAnnotatedOwnerType(); |
| Asserts.assertEquals(outer.getType(), ((Class<?>)f.getGenericType()).getEnclosingClass()); |
| Asserts.assertEquals(outer.getAnnotation(TA.class).value(), "raw"); |
| Asserts.assertTrue(outer.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + outer.getAnnotations().length); |
| } |
| |
| public static void testNonGeneric() throws Exception { |
| Field f = GetAnnotatedOwnerType.class.getField("nonGeneric"); |
| |
| // make sure inner is correctly annotated |
| AnnotatedType inner = f.getAnnotatedType(); |
| Asserts.assertEquals(inner.getAnnotation(TB.class).value(), "non-generic"); |
| Asserts.assertTrue(inner.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + inner.getAnnotations().length); |
| |
| // make sure owner is correctly annotated, on the correct type |
| AnnotatedType outer = inner.getAnnotatedOwnerType(); |
| Asserts.assertEquals(outer.getType(), ((Class<?>)f.getGenericType()).getEnclosingClass()); |
| Asserts.assertEquals(outer.getAnnotation(TA.class).value(), "non-generic"); |
| Asserts.assertTrue(outer.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + outer.getAnnotations().length); |
| } |
| |
| public static void testInnerGeneric() throws Exception { |
| Field f = GetAnnotatedOwnerType.class.getField("innerGeneric"); |
| |
| // make sure inner is correctly annotated |
| AnnotatedType inner = f.getAnnotatedType(); |
| Asserts.assertEquals(inner.getAnnotation(TB.class).value(), "generic"); |
| Asserts.assertTrue(inner.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + inner.getAnnotations().length); |
| |
| // make sure owner is correctly annotated, on the correct type |
| AnnotatedType outer = inner.getAnnotatedOwnerType(); |
| Asserts.assertEquals(outer.getType(), ((ParameterizedType) f.getGenericType()).getOwnerType()); |
| Asserts.assertEquals(outer.getAnnotation(TA.class).value(), "non-generic"); |
| Asserts.assertTrue(outer.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + outer.getAnnotations().length); |
| } |
| |
| public static void testInnerRaw() throws Exception { |
| Field f = GetAnnotatedOwnerType.class.getField("innerRaw"); |
| |
| // make sure inner is correctly annotated |
| AnnotatedType inner = f.getAnnotatedType(); |
| Asserts.assertEquals(inner.getAnnotation(TB.class).value(), "raw"); |
| Asserts.assertTrue(inner.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + inner.getAnnotations().length); |
| |
| // make sure owner is correctly annotated, on the correct type |
| AnnotatedType outer = inner.getAnnotatedOwnerType(); |
| Asserts.assertEquals(outer.getType(), ((Class<?>)f.getGenericType()).getEnclosingClass()); |
| Asserts.assertEquals(outer.getAnnotation(TA.class).value(), "non-generic"); |
| Asserts.assertTrue(outer.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + outer.getAnnotations().length); |
| } |
| |
| public static void testLocalClass() throws Exception { |
| class ALocalClass {} |
| class OneMore { |
| public @TA("null") ALocalClass c; |
| } |
| testNegative(OneMore.class.getField("c").getAnnotatedType(), "Local class should return null"); |
| } |
| |
| public static void testAnonymousClass() throws Exception { |
| testNegative(GetAnnotatedOwnerType.class.getField("anonymous").getAnnotatedType(), |
| "Anonymous class should return null"); |
| } |
| |
| public static void testArray() throws Exception { |
| AnnotatedType t = GetAnnotatedOwnerType.class.getField("dummy").getAnnotatedType(); |
| Asserts.assertTrue((t instanceof AnnotatedArrayType), |
| "Was expecting an AnnotatedArrayType " + t); |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| } |
| |
| public static void testWildcard() throws Exception { |
| AnnotatedType tt = GetAnnotatedOwnerType.class.getField("wildcard").getAnnotatedType(); |
| AnnotatedType t = ((AnnotatedParameterizedType)tt).getAnnotatedActualTypeArguments()[0]; |
| Asserts.assertTrue((t instanceof AnnotatedWildcardType), |
| "Was expecting an AnnotatedWildcardType " + t); |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| } |
| |
| public static void testTypeParameter() throws Exception { |
| AnnotatedType t = GetAnnotatedOwnerType.class.getField("tv").getAnnotatedType(); |
| Asserts.assertTrue((t instanceof AnnotatedTypeVariable), |
| "Was expecting an AnnotatedTypeVariable " + t); |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| } |
| |
| public static void testTypeArgument() throws Exception { |
| AnnotatedType tt = GetAnnotatedOwnerType.class.getField("typeArgument").getAnnotatedType(); |
| Asserts.assertEquals(tt.getAnnotation(TA.class).value(), "bad"); |
| Asserts.assertTrue(tt.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + tt.getAnnotations().length); |
| |
| // make sure inner is correctly annotated |
| AnnotatedType inner = ((AnnotatedParameterizedType)tt).getAnnotatedActualTypeArguments()[0]; |
| Asserts.assertEquals(inner.getAnnotation(TB.class).value(), "tb"); |
| Asserts.assertTrue(inner.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + inner.getAnnotations().length); |
| |
| // make sure owner is correctly annotated |
| AnnotatedType outer = inner.getAnnotatedOwnerType(); |
| Asserts.assertEquals(outer.getAnnotation(TA.class).value(), "good"); |
| Asserts.assertTrue(outer.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + outer.getAnnotations().length); |
| } |
| |
| public static void testComplicated() throws Exception { |
| Field f = GetAnnotatedOwnerType.class.getField("complicated"); |
| |
| // Outermost level |
| AnnotatedType t = f.getAnnotatedType(); |
| Asserts.assertTrue((t instanceof AnnotatedArrayType), |
| "Was expecting an AnnotatedArrayType " + t); |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| Asserts.assertTrue(t.getAnnotations().length == 0, "expecting zero annotation, got: " |
| + t.getAnnotations().length); |
| |
| // Component type |
| t = ((AnnotatedArrayType)t).getAnnotatedGenericComponentType(); |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| Asserts.assertTrue(t.getAnnotations().length == 0, "expecting zero annotation, got: " |
| + t.getAnnotations().length); |
| |
| // Type arg GetAnnotatedOwnerType<String>...D<Number> |
| t = ((AnnotatedParameterizedType)t).getAnnotatedActualTypeArguments()[0]; |
| Asserts.assertTrue(t.getAnnotations().length == 0, "expecting zero annotation, got: " |
| + t.getAnnotations().length); |
| |
| // C<Class<?>, ? extends ...> |
| t = t.getAnnotatedOwnerType(); |
| Asserts.assertTrue(t.getAnnotations().length == 0, "expecting zero annotation, got: " |
| + t.getAnnotations().length); |
| |
| // ? extends |
| t = ((AnnotatedParameterizedType)t).getAnnotatedActualTypeArguments()[1]; |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| Asserts.assertTrue(t.getAnnotations().length == 0, "expecting zero annotation, got: " |
| + t.getAnnotations().length); |
| |
| // @TA("complicated") Exception |
| t = ((AnnotatedWildcardType)t).getAnnotatedUpperBounds()[0]; |
| testNegative(t, "" + t + " should not have an annotated owner type"); |
| Asserts.assertEquals(t.getAnnotation(TA.class).value(), "complicated"); |
| Asserts.assertTrue(t.getAnnotations().length == 1, "expecting one (1) annotation, got: " |
| + t.getAnnotations().length); |
| } |
| |
| private static void testNegative(AnnotatedType t, String msg) { |
| Asserts.assertNull(t.getAnnotatedOwnerType(), msg); |
| } |
| |
| public class Nested<AlsoDummy> {} |
| public class B { |
| public class C<R, S> { |
| public class D<T> { |
| } |
| } |
| } |
| |
| @Target(ElementType.TYPE_USE) |
| @Retention(RetentionPolicy.RUNTIME) |
| public @interface TA { |
| String value(); |
| } |
| |
| @Target(ElementType.TYPE_USE) |
| @Retention(RetentionPolicy.RUNTIME) |
| public @interface TB { |
| String value(); |
| } |
| } |
| |
| class GetAnnotatedOwnerTypeAuxilliary { |
| class Inner {} |
| |
| class InnerGeneric<Dummy> {} |
| } |