From 5459304a4b5e8dc6ce08ff8d9e09228ab7d72659 Mon Sep 17 00:00:00 2001 From: Juergen Hoeller Date: Tue, 10 Oct 2023 21:54:58 +0200 Subject: [PATCH] Re-introduce support for annotation declarations with self references Closes gh-31400 --- .../annotation/AnnotationTypeMapping.java | 9 +- .../KotlinMergedAnnotationsTests.kt | 86 +++++++++++++++++++ .../core/annotation/PersonWithAlias.kt | 31 +++++++ .../core/annotation/PersonWithoutAlias.kt | 29 +++++++ 4 files changed, 152 insertions(+), 3 deletions(-) create mode 100644 spring-core/src/test/kotlin/org/springframework/core/annotation/KotlinMergedAnnotationsTests.kt create mode 100644 spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithAlias.kt create mode 100644 spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithoutAlias.kt diff --git a/spring-core/src/main/java/org/springframework/core/annotation/AnnotationTypeMapping.java b/spring-core/src/main/java/org/springframework/core/annotation/AnnotationTypeMapping.java index 05039aa8d3f9..794aa33feb49 100644 --- a/spring-core/src/main/java/org/springframework/core/annotation/AnnotationTypeMapping.java +++ b/spring-core/src/main/java/org/springframework/core/annotation/AnnotationTypeMapping.java @@ -45,6 +45,7 @@ * * @author Phillip Webb * @author Sam Brannen + * @author Juergen Hoeller * @since 5.2 * @see AnnotationTypeMappings */ @@ -402,9 +403,11 @@ private boolean computeSynthesizableFlag() { if (type.isAnnotation() || (type.isArray() && type.getComponentType().isAnnotation())) { Class annotationType = (Class) (type.isAnnotation() ? type : type.getComponentType()); - AnnotationTypeMapping mapping = AnnotationTypeMappings.forAnnotationType(annotationType).get(0); - if (mapping.isSynthesizable()) { - return true; + if (annotationType != this.annotationType) { + AnnotationTypeMapping mapping = AnnotationTypeMappings.forAnnotationType(annotationType).get(0); + if (mapping.isSynthesizable()) { + return true; + } } } } diff --git a/spring-core/src/test/kotlin/org/springframework/core/annotation/KotlinMergedAnnotationsTests.kt b/spring-core/src/test/kotlin/org/springframework/core/annotation/KotlinMergedAnnotationsTests.kt new file mode 100644 index 000000000000..2b3cc397c671 --- /dev/null +++ b/spring-core/src/test/kotlin/org/springframework/core/annotation/KotlinMergedAnnotationsTests.kt @@ -0,0 +1,86 @@ +/* + * Copyright 2002-2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.core.annotation + +import org.assertj.core.api.Assertions.assertThat +import org.junit.jupiter.api.Test + +/** + * Tests for {@link MergedAnnotations} and {@link MergedAnnotation} in Kotlin. + * + * @author Sam Brannen + * @author Juergen Hoeller + * @since 5.3.16 + */ +class KotlinMergedAnnotationsTests { + + @Test // gh-28012 + fun recursiveAnnotationWithAlias() { + val method = javaClass.getMethod("personWithAliasMethod") + + // MergedAnnotations + val mergedAnnotations = MergedAnnotations.from(method) + assertThat(mergedAnnotations.isPresent(PersonWithAlias::class.java)).isTrue(); + + // MergedAnnotation + val mergedAnnotation = MergedAnnotation.from(method.getAnnotation(PersonWithAlias::class.java)) + assertThat(mergedAnnotation).isNotNull(); + + // Synthesized Annotations + val jane = mergedAnnotation.synthesize() + assertThat(jane.value).isEqualTo("jane") + assertThat(jane.name).isEqualTo("jane") + val synthesizedFriends = jane.friends + assertThat(synthesizedFriends).hasSize(2) + + val john = synthesizedFriends[0] + assertThat(john.value).isEqualTo("john") + assertThat(john.name).isEqualTo("john") + + val sally = synthesizedFriends[1] + assertThat(sally.value).isEqualTo("sally") + assertThat(sally.name).isEqualTo("sally") + } + + @Test // gh-31400 + fun recursiveAnnotationWithoutAlias() { + val method = javaClass.getMethod("personWithoutAliasMethod") + + // MergedAnnotations + val mergedAnnotations = MergedAnnotations.from(method) + assertThat(mergedAnnotations.isPresent(PersonWithoutAlias::class.java)).isTrue(); + + // MergedAnnotation + val mergedAnnotation = MergedAnnotation.from(method.getAnnotation(PersonWithoutAlias::class.java)) + assertThat(mergedAnnotation).isNotNull(); + + // Synthesized Annotations + val jane = mergedAnnotation.synthesize() + val synthesizedFriends = jane.friends + assertThat(synthesizedFriends).hasSize(2) + } + + + @PersonWithAlias("jane", friends = [PersonWithAlias("john"), PersonWithAlias("sally")]) + fun personWithAliasMethod() { + } + + @PersonWithoutAlias("jane", friends = [PersonWithoutAlias("john"), PersonWithoutAlias("sally")]) + fun personWithoutAliasMethod() { + } + +} diff --git a/spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithAlias.kt b/spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithAlias.kt new file mode 100644 index 000000000000..249a2b83c143 --- /dev/null +++ b/spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithAlias.kt @@ -0,0 +1,31 @@ +/* + * Copyright 2002-2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.core.annotation + +@Target(AnnotationTarget.FUNCTION) +@Retention(AnnotationRetention.RUNTIME) +annotation class PersonWithAlias( + + @get:AliasFor("name") + val value: String = "", + + @get:AliasFor("value") + val name: String = "", + + vararg val friends: PersonWithAlias = [] + +) diff --git a/spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithoutAlias.kt b/spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithoutAlias.kt new file mode 100644 index 000000000000..75a918f14f74 --- /dev/null +++ b/spring-core/src/test/kotlin/org/springframework/core/annotation/PersonWithoutAlias.kt @@ -0,0 +1,29 @@ +/* + * Copyright 2002-2023 the original author or authors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * https://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.springframework.core.annotation + +@Target(AnnotationTarget.FUNCTION) +@Retention(AnnotationRetention.RUNTIME) +annotation class PersonWithoutAlias( + + val value: String = "", + + val name: String = "", + + vararg val friends: PersonWithoutAlias = [] + +)