Skip to content

Commit 8cd3b86

Browse files
committed
Merge pull request #30505 from terminux
* gh-30505: Polish "Fix ConditionalOnProperty when used in an aliased composed annotation" Fix ConditionalOnProperty when used in an aliased composed annotation Closes gh-30505
2 parents 09ec409 + 6d4f023 commit 8cd3b86

File tree

2 files changed

+68
-29
lines changed

2 files changed

+68
-29
lines changed

spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/condition/OnPropertyCondition.java

Lines changed: 8 additions & 28 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2012-2019 the original author or authors.
2+
* Copyright 2012-2022 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -17,20 +17,21 @@
1717
package org.springframework.boot.autoconfigure.condition;
1818

1919
import java.util.ArrayList;
20-
import java.util.HashMap;
2120
import java.util.List;
2221
import java.util.Map;
22+
import java.util.stream.Collectors;
2323

2424
import org.springframework.boot.autoconfigure.condition.ConditionMessage.Style;
2525
import org.springframework.context.annotation.Condition;
2626
import org.springframework.context.annotation.ConditionContext;
2727
import org.springframework.core.Ordered;
2828
import org.springframework.core.annotation.AnnotationAttributes;
29+
import org.springframework.core.annotation.MergedAnnotation;
30+
import org.springframework.core.annotation.MergedAnnotationPredicates;
2931
import org.springframework.core.annotation.Order;
3032
import org.springframework.core.env.PropertyResolver;
3133
import org.springframework.core.type.AnnotatedTypeMetadata;
3234
import org.springframework.util.Assert;
33-
import org.springframework.util.MultiValueMap;
3435
import org.springframework.util.StringUtils;
3536

3637
/**
@@ -47,8 +48,10 @@ class OnPropertyCondition extends SpringBootCondition {
4748

4849
@Override
4950
public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) {
50-
List<AnnotationAttributes> allAnnotationAttributes = annotationAttributesFromMultiValueMap(
51-
metadata.getAllAnnotationAttributes(ConditionalOnProperty.class.getName()));
51+
List<AnnotationAttributes> allAnnotationAttributes = metadata.getAnnotations()
52+
.stream(ConditionalOnProperty.class.getName())
53+
.filter(MergedAnnotationPredicates.unique(MergedAnnotation::getMetaTypes))
54+
.map(MergedAnnotation::asAnnotationAttributes).collect(Collectors.toList());
5255
List<ConditionMessage> noMatch = new ArrayList<>();
5356
List<ConditionMessage> match = new ArrayList<>();
5457
for (AnnotationAttributes annotationAttributes : allAnnotationAttributes) {
@@ -61,29 +64,6 @@ public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeM
6164
return ConditionOutcome.match(ConditionMessage.of(match));
6265
}
6366

64-
private List<AnnotationAttributes> annotationAttributesFromMultiValueMap(
65-
MultiValueMap<String, Object> multiValueMap) {
66-
List<Map<String, Object>> maps = new ArrayList<>();
67-
multiValueMap.forEach((key, value) -> {
68-
for (int i = 0; i < value.size(); i++) {
69-
Map<String, Object> map;
70-
if (i < maps.size()) {
71-
map = maps.get(i);
72-
}
73-
else {
74-
map = new HashMap<>();
75-
maps.add(map);
76-
}
77-
map.put(key, value.get(i));
78-
}
79-
});
80-
List<AnnotationAttributes> annotationAttributes = new ArrayList<>(maps.size());
81-
for (Map<String, Object> map : maps) {
82-
annotationAttributes.add(AnnotationAttributes.fromMap(map));
83-
}
84-
return annotationAttributes;
85-
}
86-
8767
private ConditionOutcome determineOutcome(AnnotationAttributes annotationAttributes, PropertyResolver resolver) {
8868
Spec spec = new Spec(annotationAttributes);
8969
List<String> missingProperties = new ArrayList<>();

spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/condition/ConditionalOnPropertyTests.java

Lines changed: 60 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2012-2021 the original author or authors.
2+
* Copyright 2012-2022 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -31,6 +31,7 @@
3131
import org.springframework.context.ConfigurableApplicationContext;
3232
import org.springframework.context.annotation.Bean;
3333
import org.springframework.context.annotation.Configuration;
34+
import org.springframework.core.annotation.AliasFor;
3435
import org.springframework.core.env.ConfigurableEnvironment;
3536
import org.springframework.core.env.StandardEnvironment;
3637

@@ -246,6 +247,31 @@ void metaAndDirectAnnotationConditionMatchesWhenBothPropertiesAreSet() {
246247
assertThat(this.context.containsBean("foo")).isTrue();
247248
}
248249

250+
@Test
251+
void metaAnnotationWithAliasConditionMatchesWhenPropertyIsSet() {
252+
load(MetaAnnotationWithAlias.class, "my.feature.enabled=true");
253+
assertThat(this.context.containsBean("foo")).isTrue();
254+
}
255+
256+
@Test
257+
void metaAndDirectAnnotationWithAliasConditionDoesNotMatchWhenOnlyMetaPropertyIsSet() {
258+
load(MetaAnnotationAndDirectAnnotationWithAlias.class, "my.feature.enabled=true");
259+
assertThat(this.context.containsBean("foo")).isFalse();
260+
}
261+
262+
@Test
263+
void metaAndDirectAnnotationWithAliasConditionDoesNotMatchWhenOnlyDirectPropertyIsSet() {
264+
load(MetaAnnotationAndDirectAnnotationWithAlias.class, "my.other.feature.enabled=true");
265+
assertThat(this.context.containsBean("foo")).isFalse();
266+
}
267+
268+
@Test
269+
void metaAndDirectAnnotationWithAliasConditionMatchesWhenBothPropertiesAreSet() {
270+
load(MetaAnnotationAndDirectAnnotationWithAlias.class, "my.feature.enabled=true",
271+
"my.other.feature.enabled=true");
272+
assertThat(this.context.containsBean("foo")).isTrue();
273+
}
274+
249275
private void load(Class<?> config, String... environment) {
250276
TestPropertyValues.of(environment).applyTo(this.environment);
251277
this.context = new SpringApplicationBuilder(config).environment(this.environment).web(WebApplicationType.NONE)
@@ -416,4 +442,37 @@ String foo() {
416442

417443
}
418444

445+
@Configuration(proxyBeanMethods = false)
446+
@ConditionalOnMyFeatureWithAlias("my.feature")
447+
static class MetaAnnotationWithAlias {
448+
449+
@Bean
450+
String foo() {
451+
return "foo";
452+
}
453+
454+
}
455+
456+
@Configuration(proxyBeanMethods = false)
457+
@ConditionalOnMyFeatureWithAlias("my.feature")
458+
@ConditionalOnProperty(prefix = "my.other.feature", name = "enabled", havingValue = "true")
459+
static class MetaAnnotationAndDirectAnnotationWithAlias {
460+
461+
@Bean
462+
String foo() {
463+
return "foo";
464+
}
465+
466+
}
467+
468+
@Retention(RetentionPolicy.RUNTIME)
469+
@Target({ ElementType.TYPE, ElementType.METHOD })
470+
@ConditionalOnProperty(name = "enabled", havingValue = "true")
471+
@interface ConditionalOnMyFeatureWithAlias {
472+
473+
@AliasFor(annotation = ConditionalOnProperty.class, attribute = "prefix")
474+
String value();
475+
476+
}
477+
419478
}

0 commit comments

Comments
 (0)