Polish 'Support programmatic lazy-int exclusion'

See gh-16615
This commit is contained in:
Phillip Webb 2019-09-26 21:14:46 -07:00
parent 0f26f4d6e2
commit 3ffc5f2a30
5 changed files with 171 additions and 112 deletions

View File

@ -1,53 +0,0 @@
/*
* Copyright 2012-2018 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.boot;
import java.util.function.Predicate;
/**
* This predicate can be implemented by downstream projects to customize the behavior of
* the {@link LazyInitializationBeanFactoryPostProcessor}.
*
* <P>
* There are edge cases (such as in DSLs that dynamically create additional beans) in
* which it is not easy to explicitly exclude a class from the lazy-loading behavior.
* Adding an instance of this predicate to the application context can be used for these
* edge cases.
* <P>
* Returning "true" from this predicate will exclude a class from the lazy-loading
* process.
*
* <P>
* Example:
* <P>
* <pre>
* {@code
*
* &#64;Bean
* public static EagerLoadingBeanDefinitionPredicate eagerLoadingBeanDefinitionPredicate() {
* return IntegrationFlow.class::isAssignableFrom;
* }}</pre>
*
* WARNING: Beans of this type will be instantiated very early in the spring application
* life cycle.
*
* @author Tyler Van Gorder
* @since 2.2.0
*/
public interface EagerLoadingBeanDefinitionPredicate extends Predicate<Class<?>> {
}

View File

@ -16,11 +16,10 @@
package org.springframework.boot;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Collection;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.NoSuchBeanDefinitionException;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
@ -28,71 +27,61 @@ import org.springframework.beans.factory.support.AbstractBeanDefinition;
import org.springframework.core.Ordered;
/**
* {@link BeanFactoryPostProcessor} to set the lazy attribute on bean definition.
*
* <P>
* This processor will not touch a bean definition that has already had its "lazy" flag
* explicitly set to "false".
*
* <P>
* There are edge cases in which it is not easy to explicitly set the "lazy" flag to
* "false" (such as in DSLs that dynamically create additional beans) and therefore this
* class uses a customizer strategy that allows downstream projects to contribute
* predicates which impact if a class is considered for lazy-loading.
*
* <P>
* Because this is a BeanFactoryPostProcessor, this class does not use dependency
* injection to collect the customizers. The post processor actually makes two passes
* through the bean definitions; the first is used to find and instantiate any
* {@link org.springframework.boot.EagerLoadingBeanDefinitionPredicate} and the second
* pass is where bean definitions are marked as lazy.
* {@link BeanFactoryPostProcessor} to set lazy-init on bean definitions that not
* {@link LazyInitializationExcludeFilter excluded} and have not already had a value
* explicitly set.
*
* @author Andy Wilkinson
* @author Madhura Bhave
* @author Tyler Van Gorder
* @author Phillip Webb
* @since 2.2.0
* @see LazyInitializationExcludeFilter
*/
public final class LazyInitializationBeanFactoryPostProcessor implements BeanFactoryPostProcessor, Ordered {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
List<EagerLoadingBeanDefinitionPredicate> eagerPredicateList = getEagerLoadingPredicatesFromContext(
beanFactory);
// Take care not to force the eager init of factory beans when getting filters
Collection<LazyInitializationExcludeFilter> filters = beanFactory
.getBeansOfType(LazyInitializationExcludeFilter.class, false, false).values();
for (String beanName : beanFactory.getBeanDefinitionNames()) {
BeanDefinition beanDefinition = beanFactory.getBeanDefinition(beanName);
if (eagerPredicateList.stream()
.anyMatch((predicate) -> predicate.test(beanFactory.getType(beanName, false)))) {
continue;
}
if (beanDefinition instanceof AbstractBeanDefinition) {
Boolean lazyInit = ((AbstractBeanDefinition) beanDefinition).getLazyInit();
if (lazyInit != null && !lazyInit) {
continue;
}
postProcess(beanFactory, filters, beanName, (AbstractBeanDefinition) beanDefinition);
}
}
}
private void postProcess(ConfigurableListableBeanFactory beanFactory,
Collection<LazyInitializationExcludeFilter> filters, String beanName,
AbstractBeanDefinition beanDefinition) {
Boolean lazyInit = beanDefinition.getLazyInit();
Class<?> beanType = getBeanType(beanFactory, beanName);
if (lazyInit == null && !isExcluded(filters, beanName, beanDefinition, beanType)) {
beanDefinition.setLazyInit(true);
}
}
/**
* This method extracts the list of
* {@link org.springframework.boot.EagerLoadingBeanDefinitionPredicate} beans from the
* bean factory. Because this method is called early in the factory life cycle, we
* take care not to force the eager initialization of factory beans.
* @param beanFactory bean factory passed into the post-processor.
* @return a list of {@link EagerLoadingBeanDefinitionPredicate} that can be used to
* customize the behavior of this processor.
*/
private List<EagerLoadingBeanDefinitionPredicate> getEagerLoadingPredicatesFromContext(
ConfigurableListableBeanFactory beanFactory) {
Map<String, EagerLoadingBeanDefinitionPredicate> eagerPredicates = beanFactory
.getBeansOfType(EagerLoadingBeanDefinitionPredicate.class, false, false);
return new ArrayList<>(eagerPredicates.values());
private Class<?> getBeanType(ConfigurableListableBeanFactory beanFactory, String beanName) {
try {
return beanFactory.getType(beanName, false);
}
catch (NoSuchBeanDefinitionException ex) {
return null;
}
}
private boolean isExcluded(Collection<LazyInitializationExcludeFilter> filters, String beanName,
AbstractBeanDefinition beanDefinition, Class<?> beanType) {
if (beanType != null) {
for (LazyInitializationExcludeFilter filter : filters) {
if (filter.isExcluded(beanName, beanDefinition, beanType)) {
return true;
}
}
}
return false;
}
@Override

View File

@ -0,0 +1,76 @@
/*
* Copyright 2012-2018 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.boot;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.AbstractBeanDefinition;
/**
* Filter that can be used to exclude beans definitions from having their
* {@link AbstractBeanDefinition#setLazyInit(boolean) lazy-int} set by the
* {@link LazyInitializationBeanFactoryPostProcessor}.
* <P>
* Primarily intended to allow downstream projects to deal with edge-cases in which it is
* not easy to support lazy-loading (such as in DSLs that dynamically create additional
* beans). Adding an instance of this filter to the application context can be used for
* these edge cases.
* <P>
* A typical example would be something like this:
* <P>
* <pre><code>
* &#64;Bean
* public static LazyInitializationExcludeFilter integrationLazyInitializationExcludeFilter() {
* return LazyInitializationExcludeFilter.forBeanTypes(IntegrationFlow.class);
* }</code></pre>
* <p>
* NOTE: Beans of this type will be instantiated very early in the spring application
* lifecycle so should generally be declared static and not have any dependencies.
*
* @author Tyler Van Gorder
* @author Philip Webb
* @since 2.2.0
*/
@FunctionalInterface
public interface LazyInitializationExcludeFilter {
/**
* Returns {@code true} if the specified bean definition should be excluded from
* having {@code lazy-int} automatically set.
* @param beanName the bean name
* @param beanDefinition the bean definition
* @param beanType the bean type
* @return {@code true} if {@code lazy-int} should not be automatically set
*/
boolean isExcluded(String beanName, BeanDefinition beanDefinition, Class<?> beanType);
/**
* Factory method that creates a filter for the given bean types.
* @param types the filtered types
* @return a new filter instance
*/
static LazyInitializationExcludeFilter forBeanTypes(Class<?>... types) {
return (beanName, beanDefinition, beanType) -> {
for (Class<?> type : types) {
if (type.isAssignableFrom(beanType)) {
return true;
}
}
return false;
};
}
}

View File

@ -0,0 +1,47 @@
/*
* Copyright 2012-2019 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.boot;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.config.BeanDefinition;
import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.Mockito.mock;
/**
* Tests for {@link LazyInitializationExcludeFilter}.
*
* @author Phillip Webb
*/
class LazyInitializationExcludeFilterTests {
@Test
void forBeanTypesMatchesTypes() {
LazyInitializationExcludeFilter filter = LazyInitializationExcludeFilter.forBeanTypes(CharSequence.class,
Number.class);
String beanName = "test";
BeanDefinition beanDefinition = mock(BeanDefinition.class);
assertThat(filter.isExcluded(beanName, beanDefinition, CharSequence.class)).isTrue();
assertThat(filter.isExcluded(beanName, beanDefinition, String.class)).isTrue();
assertThat(filter.isExcluded(beanName, beanDefinition, StringBuilder.class)).isTrue();
assertThat(filter.isExcluded(beanName, beanDefinition, Number.class)).isTrue();
assertThat(filter.isExcluded(beanName, beanDefinition, Long.class)).isTrue();
assertThat(filter.isExcluded(beanName, beanDefinition, Boolean.class)).isFalse();
}
}

View File

@ -1107,15 +1107,15 @@ class SpringApplicationTests {
}
@Test
void lazyInitializationShouldNotApplyToBeansThatAreExplicitlyNotLazy() {
void lazyInitializationIgnoresBeansThatAreExplicitlyNotLazy() {
assertThat(new SpringApplication(NotLazyInitializationConfig.class)
.run("--spring.main.web-application-type=none", "--spring.main.lazy-initialization=true")
.getBean(AtomicInteger.class)).hasValue(1);
}
@Test
void lazyInitializationShouldNotApplyToBeansThatMatchPredicate() {
assertThat(new SpringApplication(NotLazyInitializationPredicateConfig.class)
void lazyInitializationIgnoresLazyInitializationExcludeFilteredBeans() {
assertThat(new SpringApplication(LazyInitializationExcludeFilterConfig.class)
.run("--spring.main.web-application-type=none", "--spring.main.lazy-initialization=true")
.getBean(AtomicInteger.class)).hasValue(1);
}
@ -1430,7 +1430,7 @@ class SpringApplicationTests {
}
@Configuration(proxyBeanMethods = false)
static class NotLazyInitializationPredicateConfig {
static class LazyInitializationExcludeFilterConfig {
@Bean
AtomicInteger counter() {
@ -1443,16 +1443,16 @@ class SpringApplicationTests {
}
@Bean
static EagerLoadingBeanDefinitionPredicate eagerLoadingBeanDefinitionPredicate() {
return NotLazyBean.class::isAssignableFrom;
static LazyInitializationExcludeFilter lazyInitializationExcludeFilter() {
return LazyInitializationExcludeFilter.forBeanTypes(NotLazyBean.class);
}
static class NotLazyBean {
}
NotLazyBean(AtomicInteger counter) {
counter.getAndIncrement();
}
static class NotLazyBean {
NotLazyBean(AtomicInteger counter) {
counter.getAndIncrement();
}
}