Browse Source
Add SharedMetadataReaderFactoryContextInitializer to ensure that a shared caching MetadataReaderFactory is used between configuration classes and auto-configure sorting. Fixes gh-4993pull/5060/head
10 changed files with 345 additions and 17 deletions
@ -0,0 +1,145 @@
@@ -0,0 +1,145 @@
|
||||
/* |
||||
* Copyright 2012-2015 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 |
||||
* |
||||
* http://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.autoconfigure; |
||||
|
||||
import org.springframework.beans.BeansException; |
||||
import org.springframework.beans.factory.BeanClassLoaderAware; |
||||
import org.springframework.beans.factory.FactoryBean; |
||||
import org.springframework.beans.factory.NoSuchBeanDefinitionException; |
||||
import org.springframework.beans.factory.config.BeanDefinition; |
||||
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; |
||||
import org.springframework.beans.factory.config.RuntimeBeanReference; |
||||
import org.springframework.beans.factory.support.BeanDefinitionRegistry; |
||||
import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor; |
||||
import org.springframework.beans.factory.support.RootBeanDefinition; |
||||
import org.springframework.boot.type.classreading.ConcurrentReferenceCachingMetadataReaderFactory; |
||||
import org.springframework.context.ApplicationContextInitializer; |
||||
import org.springframework.context.ApplicationListener; |
||||
import org.springframework.context.ConfigurableApplicationContext; |
||||
import org.springframework.context.annotation.AnnotationConfigUtils; |
||||
import org.springframework.context.annotation.ConfigurationClassPostProcessor; |
||||
import org.springframework.context.event.ContextRefreshedEvent; |
||||
import org.springframework.core.Ordered; |
||||
import org.springframework.core.PriorityOrdered; |
||||
import org.springframework.core.type.classreading.CachingMetadataReaderFactory; |
||||
import org.springframework.core.type.classreading.MetadataReaderFactory; |
||||
|
||||
/** |
||||
* {@link ApplicationContextInitializer} to create a shared |
||||
* {@link CachingMetadataReaderFactory} between the |
||||
* {@link ConfigurationClassPostProcessor} and Spring Boot. |
||||
* |
||||
* @author Phillip Webb |
||||
* @since 1.4.0 |
||||
*/ |
||||
class SharedMetadataReaderFactoryContextInitializer |
||||
implements ApplicationContextInitializer<ConfigurableApplicationContext> { |
||||
|
||||
public static final String BEAN_NAME = "org.springframework.boot.autoconfigure." |
||||
+ "internalCachingMetadataReaderFactory"; |
||||
|
||||
@Override |
||||
public void initialize(ConfigurableApplicationContext applicationContext) { |
||||
applicationContext.addBeanFactoryPostProcessor( |
||||
new CachingMetadataReaderFactoryPostProcessor()); |
||||
} |
||||
|
||||
/** |
||||
* {@link BeanDefinitionRegistryPostProcessor} to register the |
||||
* {@link CachingMetadataReaderFactory} and configure the |
||||
* {@link ConfigurationClassPostProcessor}. |
||||
*/ |
||||
private static class CachingMetadataReaderFactoryPostProcessor |
||||
implements BeanDefinitionRegistryPostProcessor, PriorityOrdered { |
||||
|
||||
@Override |
||||
public int getOrder() { |
||||
// Must happen before the ConfigurationClassPostProcessor is created
|
||||
return Ordered.HIGHEST_PRECEDENCE; |
||||
} |
||||
|
||||
@Override |
||||
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) |
||||
throws BeansException { |
||||
} |
||||
|
||||
@Override |
||||
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) |
||||
throws BeansException { |
||||
register(registry); |
||||
configureConfigurationClassPostProcessor(registry); |
||||
} |
||||
|
||||
private void register(BeanDefinitionRegistry registry) { |
||||
RootBeanDefinition definition = new RootBeanDefinition( |
||||
SharedMetadataReaderFactoryBean.class); |
||||
registry.registerBeanDefinition(BEAN_NAME, definition); |
||||
} |
||||
|
||||
private void configureConfigurationClassPostProcessor( |
||||
BeanDefinitionRegistry registry) { |
||||
try { |
||||
BeanDefinition definition = registry.getBeanDefinition( |
||||
AnnotationConfigUtils.CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME); |
||||
definition.getPropertyValues().add("metadataReaderFactory", |
||||
new RuntimeBeanReference(BEAN_NAME)); |
||||
} |
||||
catch (NoSuchBeanDefinitionException ex) { |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
/** |
||||
* {@link FactoryBean} to create the shared {@link MetadataReaderFactory}. |
||||
*/ |
||||
static class SharedMetadataReaderFactoryBean |
||||
implements FactoryBean<ConcurrentReferenceCachingMetadataReaderFactory>, |
||||
BeanClassLoaderAware, ApplicationListener<ContextRefreshedEvent> { |
||||
|
||||
private ConcurrentReferenceCachingMetadataReaderFactory metadataReaderFactory; |
||||
|
||||
@Override |
||||
public void setBeanClassLoader(ClassLoader classLoader) { |
||||
this.metadataReaderFactory = new ConcurrentReferenceCachingMetadataReaderFactory( |
||||
classLoader); |
||||
} |
||||
|
||||
@Override |
||||
public ConcurrentReferenceCachingMetadataReaderFactory getObject() |
||||
throws Exception { |
||||
return this.metadataReaderFactory; |
||||
} |
||||
|
||||
@Override |
||||
public Class<?> getObjectType() { |
||||
return CachingMetadataReaderFactory.class; |
||||
} |
||||
|
||||
@Override |
||||
public boolean isSingleton() { |
||||
return true; |
||||
} |
||||
|
||||
@Override |
||||
public void onApplicationEvent(ContextRefreshedEvent event) { |
||||
this.metadataReaderFactory.clearCache(); |
||||
} |
||||
|
||||
} |
||||
|
||||
} |
||||
@ -0,0 +1,99 @@
@@ -0,0 +1,99 @@
|
||||
/* |
||||
* Copyright 2012-2015 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 |
||||
* |
||||
* http://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.type.classreading; |
||||
|
||||
import java.io.IOException; |
||||
import java.util.Map; |
||||
|
||||
import org.springframework.core.io.Resource; |
||||
import org.springframework.core.io.ResourceLoader; |
||||
import org.springframework.core.type.classreading.CachingMetadataReaderFactory; |
||||
import org.springframework.core.type.classreading.MetadataReader; |
||||
import org.springframework.core.type.classreading.MetadataReaderFactory; |
||||
import org.springframework.core.type.classreading.SimpleMetadataReaderFactory; |
||||
import org.springframework.util.ConcurrentReferenceHashMap; |
||||
|
||||
/** |
||||
* Caching implementation of the {@link MetadataReaderFactory} interface backed by a |
||||
* {@link ConcurrentReferenceHashMap} , caching {@link MetadataReader} per Spring |
||||
* {@link Resource} handle (i.e. per ".class" file). |
||||
* |
||||
* @author Phillip Webb |
||||
* @since 1.4.0 |
||||
* @see CachingMetadataReaderFactory |
||||
*/ |
||||
public class ConcurrentReferenceCachingMetadataReaderFactory |
||||
extends SimpleMetadataReaderFactory { |
||||
|
||||
private final Map<Resource, MetadataReader> cache = new ConcurrentReferenceHashMap<Resource, MetadataReader>(); |
||||
|
||||
/** |
||||
* Create a new {@link ConcurrentReferenceCachingMetadataReaderFactory} instance for |
||||
* the default class loader. |
||||
*/ |
||||
public ConcurrentReferenceCachingMetadataReaderFactory() { |
||||
super(); |
||||
} |
||||
|
||||
/** |
||||
* Create a new {@link ConcurrentReferenceCachingMetadataReaderFactory} instance for |
||||
* the given resource loader. |
||||
* @param resourceLoader the Spring ResourceLoader to use (also determines the |
||||
* ClassLoader to use) |
||||
*/ |
||||
public ConcurrentReferenceCachingMetadataReaderFactory( |
||||
ResourceLoader resourceLoader) { |
||||
super(resourceLoader); |
||||
} |
||||
|
||||
/** |
||||
* Create a new {@link ConcurrentReferenceCachingMetadataReaderFactory} instance for |
||||
* the given class loader. |
||||
* @param classLoader the ClassLoader to use |
||||
*/ |
||||
public ConcurrentReferenceCachingMetadataReaderFactory(ClassLoader classLoader) { |
||||
super(classLoader); |
||||
} |
||||
|
||||
@Override |
||||
public MetadataReader getMetadataReader(Resource resource) throws IOException { |
||||
MetadataReader metadataReader = this.cache.get(resource); |
||||
if (metadataReader == null) { |
||||
metadataReader = createMetadataReader(resource); |
||||
this.cache.put(resource, metadataReader); |
||||
} |
||||
return metadataReader; |
||||
} |
||||
|
||||
/** |
||||
* Create the meta-data reader. |
||||
* @param resource the source resource. |
||||
* @return the meta-data reader |
||||
* @throws IOException on error |
||||
*/ |
||||
protected MetadataReader createMetadataReader(Resource resource) throws IOException { |
||||
return super.getMetadataReader(resource); |
||||
} |
||||
|
||||
/** |
||||
* Clear the entire MetadataReader cache, removing all cached class metadata. |
||||
*/ |
||||
public void clearCache() { |
||||
this.cache.clear(); |
||||
} |
||||
|
||||
} |
||||
@ -0,0 +1,73 @@
@@ -0,0 +1,73 @@
|
||||
/* |
||||
* Copyright 2012-2015 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 |
||||
* |
||||
* http://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.type.classreading; |
||||
|
||||
import java.io.IOException; |
||||
|
||||
import org.junit.Test; |
||||
|
||||
import org.springframework.core.io.Resource; |
||||
import org.springframework.core.type.classreading.MetadataReader; |
||||
|
||||
import static org.hamcrest.Matchers.not; |
||||
import static org.hamcrest.Matchers.sameInstance; |
||||
import static org.junit.Assert.assertThat; |
||||
import static org.mockito.Matchers.any; |
||||
import static org.mockito.Mockito.mock; |
||||
import static org.mockito.Mockito.spy; |
||||
import static org.mockito.Mockito.times; |
||||
import static org.mockito.Mockito.verify; |
||||
|
||||
/** |
||||
* Tests for {@link ConcurrentReferenceCachingMetadataReaderFactory}. |
||||
* |
||||
* @author Phillip Webb |
||||
*/ |
||||
public class ConcurrentReferenceCachingMetadataReaderFactoryTests { |
||||
|
||||
@Test |
||||
public void getMetadataReaderUsesCache() throws Exception { |
||||
TestConcurrentReferenceCachingMetadataReaderFactory factory = spy( |
||||
new TestConcurrentReferenceCachingMetadataReaderFactory()); |
||||
MetadataReader metadataReader1 = factory.getMetadataReader(getClass().getName()); |
||||
MetadataReader metadataReader2 = factory.getMetadataReader(getClass().getName()); |
||||
assertThat(metadataReader1, sameInstance(metadataReader2)); |
||||
verify(factory, times(1)).createMetadataReader((Resource) any()); |
||||
} |
||||
|
||||
@Test |
||||
public void clearResetsCache() throws Exception { |
||||
TestConcurrentReferenceCachingMetadataReaderFactory factory = spy( |
||||
new TestConcurrentReferenceCachingMetadataReaderFactory()); |
||||
MetadataReader metadataReader1 = factory.getMetadataReader(getClass().getName()); |
||||
factory.clearCache(); |
||||
MetadataReader metadataReader2 = factory.getMetadataReader(getClass().getName()); |
||||
assertThat(metadataReader1, not(sameInstance(metadataReader2))); |
||||
verify(factory, times(2)).createMetadataReader((Resource) any()); |
||||
} |
||||
|
||||
private static class TestConcurrentReferenceCachingMetadataReaderFactory |
||||
extends ConcurrentReferenceCachingMetadataReaderFactory { |
||||
|
||||
@Override |
||||
public MetadataReader createMetadataReader(Resource resource) throws IOException { |
||||
return mock(MetadataReader.class); |
||||
} |
||||
|
||||
} |
||||
|
||||
} |
||||
Loading…
Reference in new issue