Introduce fallback flag and annotation (as companion to primary)
Closes gh-26241
This commit is contained in:
parent
c077805761
commit
480051a21c
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2020 the original author or authors.
|
* Copyright 2002-2024 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -178,6 +178,7 @@ public interface BeanDefinition extends AttributeAccessor, BeanMetadataElement {
|
||||||
* Set whether this bean is a primary autowire candidate.
|
* Set whether this bean is a primary autowire candidate.
|
||||||
* <p>If this value is {@code true} for exactly one bean among multiple
|
* <p>If this value is {@code true} for exactly one bean among multiple
|
||||||
* matching candidates, it will serve as a tie-breaker.
|
* matching candidates, it will serve as a tie-breaker.
|
||||||
|
* @see #setFallback
|
||||||
*/
|
*/
|
||||||
void setPrimary(boolean primary);
|
void setPrimary(boolean primary);
|
||||||
|
|
||||||
|
|
@ -186,6 +187,21 @@ public interface BeanDefinition extends AttributeAccessor, BeanMetadataElement {
|
||||||
*/
|
*/
|
||||||
boolean isPrimary();
|
boolean isPrimary();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set whether this bean is a fallback autowire candidate.
|
||||||
|
* <p>If this value is {@code true} for all beans but one among multiple
|
||||||
|
* matching candidates, the remaining bean will be selected.
|
||||||
|
* @since 6.2
|
||||||
|
* @see #setPrimary
|
||||||
|
*/
|
||||||
|
void setFallback(boolean fallback);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return whether this bean is a fallback autowire candidate.
|
||||||
|
* @since 6.2
|
||||||
|
*/
|
||||||
|
boolean isFallback();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Specify the factory bean to use, if any.
|
* Specify the factory bean to use, if any.
|
||||||
* This the name of the bean to call the specified factory method on.
|
* This the name of the bean to call the specified factory method on.
|
||||||
|
|
|
||||||
|
|
@ -189,6 +189,8 @@ public abstract class AbstractBeanDefinition extends BeanMetadataAttributeAccess
|
||||||
|
|
||||||
private boolean primary = false;
|
private boolean primary = false;
|
||||||
|
|
||||||
|
private boolean fallback = false;
|
||||||
|
|
||||||
private final Map<String, AutowireCandidateQualifier> qualifiers = new LinkedHashMap<>();
|
private final Map<String, AutowireCandidateQualifier> qualifiers = new LinkedHashMap<>();
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
|
|
@ -288,6 +290,7 @@ public abstract class AbstractBeanDefinition extends BeanMetadataAttributeAccess
|
||||||
setAutowireCandidate(originalAbd.isAutowireCandidate());
|
setAutowireCandidate(originalAbd.isAutowireCandidate());
|
||||||
setDefaultCandidate(originalAbd.isDefaultCandidate());
|
setDefaultCandidate(originalAbd.isDefaultCandidate());
|
||||||
setPrimary(originalAbd.isPrimary());
|
setPrimary(originalAbd.isPrimary());
|
||||||
|
setFallback(originalAbd.isFallback());
|
||||||
copyQualifiersFrom(originalAbd);
|
copyQualifiersFrom(originalAbd);
|
||||||
setInstanceSupplier(originalAbd.getInstanceSupplier());
|
setInstanceSupplier(originalAbd.getInstanceSupplier());
|
||||||
setNonPublicAccessAllowed(originalAbd.isNonPublicAccessAllowed());
|
setNonPublicAccessAllowed(originalAbd.isNonPublicAccessAllowed());
|
||||||
|
|
@ -365,6 +368,7 @@ public abstract class AbstractBeanDefinition extends BeanMetadataAttributeAccess
|
||||||
setAutowireCandidate(otherAbd.isAutowireCandidate());
|
setAutowireCandidate(otherAbd.isAutowireCandidate());
|
||||||
setDefaultCandidate(otherAbd.isDefaultCandidate());
|
setDefaultCandidate(otherAbd.isDefaultCandidate());
|
||||||
setPrimary(otherAbd.isPrimary());
|
setPrimary(otherAbd.isPrimary());
|
||||||
|
setFallback(otherAbd.isFallback());
|
||||||
copyQualifiersFrom(otherAbd);
|
copyQualifiersFrom(otherAbd);
|
||||||
setInstanceSupplier(otherAbd.getInstanceSupplier());
|
setInstanceSupplier(otherAbd.getInstanceSupplier());
|
||||||
setNonPublicAccessAllowed(otherAbd.isNonPublicAccessAllowed());
|
setNonPublicAccessAllowed(otherAbd.isNonPublicAccessAllowed());
|
||||||
|
|
@ -742,6 +746,7 @@ public abstract class AbstractBeanDefinition extends BeanMetadataAttributeAccess
|
||||||
* Set whether this bean is a primary autowire candidate.
|
* Set whether this bean is a primary autowire candidate.
|
||||||
* <p>Default is {@code false}. If this value is {@code true} for exactly one
|
* <p>Default is {@code false}. If this value is {@code true} for exactly one
|
||||||
* bean among multiple matching candidates, it will serve as a tie-breaker.
|
* bean among multiple matching candidates, it will serve as a tie-breaker.
|
||||||
|
* @see #setFallback
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void setPrimary(boolean primary) {
|
public void setPrimary(boolean primary) {
|
||||||
|
|
@ -756,6 +761,25 @@ public abstract class AbstractBeanDefinition extends BeanMetadataAttributeAccess
|
||||||
return this.primary;
|
return this.primary;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set whether this bean is a fallback autowire candidate.
|
||||||
|
* <p>Default is {@code false}. If this value is {@code true} for all beans but
|
||||||
|
* one among multiple matching candidates, the remaining bean will be selected.
|
||||||
|
* @since 6.2
|
||||||
|
* @see #setPrimary
|
||||||
|
*/
|
||||||
|
public void setFallback(boolean fallback) {
|
||||||
|
this.fallback = fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return whether this bean is a fallback autowire candidate.
|
||||||
|
* @since 6.2
|
||||||
|
*/
|
||||||
|
public boolean isFallback() {
|
||||||
|
return this.fallback;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register a qualifier to be used for autowire candidate resolution,
|
* Register a qualifier to be used for autowire candidate resolution,
|
||||||
* keyed by the qualifier's type name.
|
* keyed by the qualifier's type name.
|
||||||
|
|
|
||||||
|
|
@ -1796,6 +1796,7 @@ public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFacto
|
||||||
@Nullable
|
@Nullable
|
||||||
protected String determinePrimaryCandidate(Map<String, Object> candidates, Class<?> requiredType) {
|
protected String determinePrimaryCandidate(Map<String, Object> candidates, Class<?> requiredType) {
|
||||||
String primaryBeanName = null;
|
String primaryBeanName = null;
|
||||||
|
// First pass: identify unique primary candidate
|
||||||
for (Map.Entry<String, Object> entry : candidates.entrySet()) {
|
for (Map.Entry<String, Object> entry : candidates.entrySet()) {
|
||||||
String candidateBeanName = entry.getKey();
|
String candidateBeanName = entry.getKey();
|
||||||
Object beanInstance = entry.getValue();
|
Object beanInstance = entry.getValue();
|
||||||
|
|
@ -1816,6 +1817,19 @@ public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFacto
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Second pass: identify unique non-fallback candidate
|
||||||
|
if (primaryBeanName == null) {
|
||||||
|
for (Map.Entry<String, Object> entry : candidates.entrySet()) {
|
||||||
|
String candidateBeanName = entry.getKey();
|
||||||
|
Object beanInstance = entry.getValue();
|
||||||
|
if (!isFallback(candidateBeanName, beanInstance)) {
|
||||||
|
if (primaryBeanName != null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
primaryBeanName = candidateBeanName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
return primaryBeanName;
|
return primaryBeanName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1878,6 +1892,23 @@ public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFacto
|
||||||
parent.isPrimary(transformedBeanName, beanInstance));
|
parent.isPrimary(transformedBeanName, beanInstance));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return whether the bean definition for the given bean name has been
|
||||||
|
* marked as a fallback bean.
|
||||||
|
* @param beanName the name of the bean
|
||||||
|
* @param beanInstance the corresponding bean instance (can be {@code null})
|
||||||
|
* @return whether the given bean qualifies as fallback
|
||||||
|
* @since 6.2
|
||||||
|
*/
|
||||||
|
protected boolean isFallback(String beanName, Object beanInstance) {
|
||||||
|
String transformedBeanName = transformedBeanName(beanName);
|
||||||
|
if (containsBeanDefinition(transformedBeanName)) {
|
||||||
|
return getMergedLocalBeanDefinition(transformedBeanName).isFallback();
|
||||||
|
}
|
||||||
|
return (getParentBeanFactory() instanceof DefaultListableBeanFactory parent &&
|
||||||
|
parent.isFallback(transformedBeanName, beanInstance));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the priority assigned for the given bean instance by
|
* Return the priority assigned for the given bean instance by
|
||||||
* the {@code jakarta.annotation.Priority} annotation.
|
* the {@code jakarta.annotation.Priority} annotation.
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2023 the original author or authors.
|
* Copyright 2002-2024 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -246,6 +246,9 @@ public abstract class AnnotationConfigUtils {
|
||||||
if (metadata.isAnnotated(Primary.class.getName())) {
|
if (metadata.isAnnotated(Primary.class.getName())) {
|
||||||
abd.setPrimary(true);
|
abd.setPrimary(true);
|
||||||
}
|
}
|
||||||
|
if (metadata.isAnnotated(Fallback.class.getName())) {
|
||||||
|
abd.setFallback(true);
|
||||||
|
}
|
||||||
AnnotationAttributes dependsOn = attributesFor(metadata, DependsOn.class);
|
AnnotationAttributes dependsOn = attributesFor(metadata, DependsOn.class);
|
||||||
if (dependsOn != null) {
|
if (dependsOn != null) {
|
||||||
abd.setDependsOn(dependsOn.getStringArray("value"));
|
abd.setDependsOn(dependsOn.getStringArray("value"));
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,44 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2024 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.context.annotation;
|
||||||
|
|
||||||
|
import java.lang.annotation.Documented;
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Indicates that a bean qualifies as a fallback autowire candidate.
|
||||||
|
* This is a companion and alternative to the {@link Primary} annotation.
|
||||||
|
*
|
||||||
|
* <p>If all beans but one among multiple matching candidates are marked
|
||||||
|
* as a fallback, the remaining bean will be selected.
|
||||||
|
*
|
||||||
|
* @author Juergen Hoeller
|
||||||
|
* @since 6.2
|
||||||
|
* @see Primary
|
||||||
|
* @see Lazy
|
||||||
|
* @see Bean
|
||||||
|
* @see org.springframework.beans.factory.config.BeanDefinition#setFallback
|
||||||
|
*/
|
||||||
|
@Target({ElementType.TYPE, ElementType.METHOD})
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Documented
|
||||||
|
public @interface Fallback {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2016 the original author or authors.
|
* Copyright 2002-2024 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -77,10 +77,12 @@ import java.lang.annotation.Target;
|
||||||
* @author Chris Beams
|
* @author Chris Beams
|
||||||
* @author Juergen Hoeller
|
* @author Juergen Hoeller
|
||||||
* @since 3.0
|
* @since 3.0
|
||||||
|
* @see Fallback
|
||||||
* @see Lazy
|
* @see Lazy
|
||||||
* @see Bean
|
* @see Bean
|
||||||
* @see ComponentScan
|
* @see ComponentScan
|
||||||
* @see org.springframework.stereotype.Component
|
* @see org.springframework.stereotype.Component
|
||||||
|
* @see org.springframework.beans.factory.config.BeanDefinition#setPrimary
|
||||||
*/
|
*/
|
||||||
@Target({ElementType.TYPE, ElementType.METHOD})
|
@Target({ElementType.TYPE, ElementType.METHOD})
|
||||||
@Retention(RetentionPolicy.RUNTIME)
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
|
|
||||||
|
|
@ -30,7 +30,9 @@ import org.springframework.beans.testfixture.beans.TestBean;
|
||||||
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.annotation.Fallback;
|
||||||
import org.springframework.context.annotation.Lazy;
|
import org.springframework.context.annotation.Lazy;
|
||||||
|
import org.springframework.context.annotation.Primary;
|
||||||
import org.springframework.context.annotation.Scope;
|
import org.springframework.context.annotation.Scope;
|
||||||
import org.springframework.context.annotation.ScopedProxyMode;
|
import org.springframework.context.annotation.ScopedProxyMode;
|
||||||
import org.springframework.core.annotation.AliasFor;
|
import org.springframework.core.annotation.AliasFor;
|
||||||
|
|
@ -80,6 +82,26 @@ class BeanMethodQualificationTests {
|
||||||
ctx.close();
|
ctx.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void primary() {
|
||||||
|
AnnotationConfigApplicationContext ctx =
|
||||||
|
new AnnotationConfigApplicationContext(PrimaryConfig.class, StandardPojo.class);
|
||||||
|
StandardPojo pojo = ctx.getBean(StandardPojo.class);
|
||||||
|
assertThat(pojo.testBean.getName()).isEqualTo("interesting");
|
||||||
|
assertThat(pojo.testBean2.getName()).isEqualTo("boring");
|
||||||
|
ctx.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void fallback() {
|
||||||
|
AnnotationConfigApplicationContext ctx =
|
||||||
|
new AnnotationConfigApplicationContext(FallbackConfig.class, StandardPojo.class);
|
||||||
|
StandardPojo pojo = ctx.getBean(StandardPojo.class);
|
||||||
|
assertThat(pojo.testBean.getName()).isEqualTo("interesting");
|
||||||
|
assertThat(pojo.testBean2.getName()).isEqualTo("boring");
|
||||||
|
ctx.close();
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void customWithLazyResolution() {
|
void customWithLazyResolution() {
|
||||||
AnnotationConfigApplicationContext ctx =
|
AnnotationConfigApplicationContext ctx =
|
||||||
|
|
@ -201,6 +223,58 @@ class BeanMethodQualificationTests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
static class PrimaryConfig {
|
||||||
|
|
||||||
|
@Bean @Qualifier("interesting") @Primary
|
||||||
|
public static TestBean testBean1() {
|
||||||
|
return new TestBean("interesting");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean @Qualifier("interesting")
|
||||||
|
public static TestBean testBean1x() {
|
||||||
|
return new TestBean("interesting");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean @Boring @Primary
|
||||||
|
public TestBean testBean2(TestBean testBean1) {
|
||||||
|
TestBean tb = new TestBean("boring");
|
||||||
|
tb.setSpouse(testBean1);
|
||||||
|
return tb;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean @Boring
|
||||||
|
public TestBean testBean2x() {
|
||||||
|
return new TestBean("boring");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
static class FallbackConfig {
|
||||||
|
|
||||||
|
@Bean @Qualifier("interesting")
|
||||||
|
public static TestBean testBean1() {
|
||||||
|
return new TestBean("interesting");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean @Qualifier("interesting") @Fallback
|
||||||
|
public static TestBean testBean1x() {
|
||||||
|
return new TestBean("interesting");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean @Boring
|
||||||
|
public TestBean testBean2(TestBean testBean1) {
|
||||||
|
TestBean tb = new TestBean("boring");
|
||||||
|
tb.setSpouse(testBean1);
|
||||||
|
return tb;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean @Boring @Fallback
|
||||||
|
public TestBean testBean2x() {
|
||||||
|
return new TestBean("boring");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Component @Lazy
|
@Component @Lazy
|
||||||
static class StandardPojo {
|
static class StandardPojo {
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue