Skip to content

Commit f67ddea

Browse files
committed
Avoid infinite recursion in AOT processing with recursive generics
1 parent 5e08a88 commit f67ddea

File tree

2 files changed

+35
-7
lines changed

2 files changed

+35
-7
lines changed

Diff for: spring-context/src/main/java/org/springframework/validation/beanvalidation/BeanValidationBeanRegistrationAotProcessor.java

+11-7
Original file line numberDiff line numberDiff line change
@@ -104,20 +104,24 @@ public static BeanRegistrationAotContribution processAheadOfTime(RegisteredBean
104104
}
105105

106106
Class<?> beanClass = registeredBean.getBeanClass();
107+
Set<Class<?>> visitedClasses = new HashSet<>();
107108
Set<Class<?>> validatedClasses = new HashSet<>();
108109
Set<Class<? extends ConstraintValidator<?, ?>>> constraintValidatorClasses = new HashSet<>();
109110

110-
processAheadOfTime(beanClass, validatedClasses, constraintValidatorClasses);
111+
processAheadOfTime(beanClass, visitedClasses, validatedClasses, constraintValidatorClasses);
111112

112113
if (!validatedClasses.isEmpty() || !constraintValidatorClasses.isEmpty()) {
113114
return new AotContribution(validatedClasses, constraintValidatorClasses);
114115
}
115116
return null;
116117
}
117118

118-
private static void processAheadOfTime(Class<?> clazz, Collection<Class<?>> validatedClasses,
119-
Collection<Class<? extends ConstraintValidator<?, ?>>> constraintValidatorClasses) {
119+
private static void processAheadOfTime(Class<?> clazz, Set<Class<?>> visitedClasses, Set<Class<?>> validatedClasses,
120+
Set<Class<? extends ConstraintValidator<?, ?>>> constraintValidatorClasses) {
120121

122+
if (visitedClasses.add(clazz)) {
123+
return;
124+
}
121125
Assert.notNull(validator, "Validator can't be null");
122126

123127
BeanDescriptor descriptor;
@@ -149,12 +153,12 @@ else if (ex instanceof TypeNotPresentException) {
149153

150154
ReflectionUtils.doWithFields(clazz, field -> {
151155
Class<?> type = field.getType();
152-
if (Iterable.class.isAssignableFrom(type) || List.class.isAssignableFrom(type) || Optional.class.isAssignableFrom(type)) {
156+
if (Iterable.class.isAssignableFrom(type) || Optional.class.isAssignableFrom(type)) {
153157
ResolvableType resolvableType = ResolvableType.forField(field);
154158
Class<?> genericType = resolvableType.getGeneric(0).toClass();
155159
if (shouldProcess(genericType)) {
156160
validatedClasses.add(clazz);
157-
processAheadOfTime(genericType, validatedClasses, constraintValidatorClasses);
161+
processAheadOfTime(genericType, visitedClasses, validatedClasses, constraintValidatorClasses);
158162
}
159163
}
160164
if (Map.class.isAssignableFrom(type)) {
@@ -163,11 +167,11 @@ else if (ex instanceof TypeNotPresentException) {
163167
Class<?> valueGenericType = resolvableType.getGeneric(1).toClass();
164168
if (shouldProcess(keyGenericType)) {
165169
validatedClasses.add(clazz);
166-
processAheadOfTime(keyGenericType, validatedClasses, constraintValidatorClasses);
170+
processAheadOfTime(keyGenericType, visitedClasses, validatedClasses, constraintValidatorClasses);
167171
}
168172
if (shouldProcess(valueGenericType)) {
169173
validatedClasses.add(clazz);
170-
processAheadOfTime(valueGenericType, validatedClasses, constraintValidatorClasses);
174+
processAheadOfTime(valueGenericType, visitedClasses, validatedClasses, constraintValidatorClasses);
171175
}
172176
}
173177
});

Diff for: spring-context/src/test/java/org/springframework/validation/beanvalidation/BeanValidationBeanRegistrationAotProcessorTests.java

+24
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,9 @@
2222
import java.lang.annotation.Target;
2323
import java.util.ArrayList;
2424
import java.util.List;
25+
import java.util.Map;
26+
import java.util.Optional;
27+
import java.util.Set;
2528

2629
import jakarta.validation.Constraint;
2730
import jakarta.validation.ConstraintValidator;
@@ -32,6 +35,8 @@
3235
import org.hibernate.validator.internal.constraintvalidators.bv.PatternValidator;
3336
import org.junit.jupiter.api.Test;
3437

38+
import org.junit.jupiter.params.ParameterizedTest;
39+
import org.junit.jupiter.params.provider.ValueSource;
3540
import org.springframework.aot.generate.GenerationContext;
3641
import org.springframework.aot.hint.MemberCategory;
3742
import org.springframework.aot.hint.predicate.RuntimeHintsPredicates;
@@ -121,6 +126,13 @@ void shouldProcessTransitiveGenericTypeLevelConstraint() {
121126
.withMemberCategory(MemberCategory.INVOKE_DECLARED_CONSTRUCTORS)).accepts(this.generationContext.getRuntimeHints());
122127
}
123128

129+
@ParameterizedTest // gh-33936
130+
@ValueSource(classes = {BeanWithIterable.class, BeanWithMap.class, BeanWithOptional.class})
131+
void shouldProcessRecursiveGenericsWithoutInfiniteRecursion(Class<?> beanClass) {
132+
process(beanClass);
133+
assertThat(this.generationContext.getRuntimeHints().reflection().typeHints()).hasSize(0);
134+
}
135+
124136
private void process(Class<?> beanClass) {
125137
BeanRegistrationAotContribution contribution = createContribution(beanClass);
126138
if (contribution != null) {
@@ -244,4 +256,16 @@ public void setExclude(List<Exclude> exclude) {
244256
}
245257
}
246258

259+
static class BeanWithIterable {
260+
private final Iterable<BeanWithIterable> beans = Set.of();
261+
}
262+
263+
static class BeanWithMap {
264+
private final Map<String, BeanWithMap> beans = Map.of();
265+
}
266+
267+
static class BeanWithOptional {
268+
private final Optional<BeanWithOptional> beans = Optional.empty();
269+
}
270+
247271
}

0 commit comments

Comments
 (0)