Browse Source
gh-33616 refactored `CachingMetadataReaderFactory` and broke the behavior as it bypassed the cache for `getMetadataReader(String className)` operations. This commit restores the original behavior. Fixes gh-35112pull/35123/head
7 changed files with 161 additions and 183 deletions
@ -0,0 +1,82 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2002-present 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.core.type.classreading; |
||||||
|
|
||||||
|
import java.io.FileNotFoundException; |
||||||
|
import java.io.IOException; |
||||||
|
|
||||||
|
import org.jspecify.annotations.Nullable; |
||||||
|
|
||||||
|
import org.springframework.core.io.DefaultResourceLoader; |
||||||
|
import org.springframework.core.io.Resource; |
||||||
|
import org.springframework.core.io.ResourceLoader; |
||||||
|
import org.springframework.util.ClassUtils; |
||||||
|
|
||||||
|
abstract class AbstractMetadataReaderFactory implements MetadataReaderFactory { |
||||||
|
|
||||||
|
private final ResourceLoader resourceLoader; |
||||||
|
|
||||||
|
|
||||||
|
public AbstractMetadataReaderFactory(@Nullable ResourceLoader resourceLoader) { |
||||||
|
this.resourceLoader = (resourceLoader != null ? resourceLoader : new DefaultResourceLoader()); |
||||||
|
} |
||||||
|
|
||||||
|
public AbstractMetadataReaderFactory(@Nullable ClassLoader classLoader) { |
||||||
|
this.resourceLoader = |
||||||
|
(classLoader != null ? new DefaultResourceLoader(classLoader) : new DefaultResourceLoader()); |
||||||
|
} |
||||||
|
|
||||||
|
public AbstractMetadataReaderFactory() { |
||||||
|
this.resourceLoader = new DefaultResourceLoader(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Return the ResourceLoader that this MetadataReaderFactory has been |
||||||
|
* constructed with. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public ResourceLoader getResourceLoader() { |
||||||
|
return this.resourceLoader; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public MetadataReader getMetadataReader(String className) throws IOException { |
||||||
|
try { |
||||||
|
String resourcePath = ResourceLoader.CLASSPATH_URL_PREFIX + |
||||||
|
ClassUtils.convertClassNameToResourcePath(className) + ClassUtils.CLASS_FILE_SUFFIX; |
||||||
|
Resource resource = this.resourceLoader.getResource(resourcePath); |
||||||
|
return getMetadataReader(resource); |
||||||
|
} |
||||||
|
catch (FileNotFoundException ex) { |
||||||
|
// Maybe an inner class name using the dot name syntax? Need to use the dollar syntax here...
|
||||||
|
// ClassUtils.forName has an equivalent check for resolution into Class references later on.
|
||||||
|
int lastDotIndex = className.lastIndexOf('.'); |
||||||
|
if (lastDotIndex != -1) { |
||||||
|
String innerClassName = |
||||||
|
className.substring(0, lastDotIndex) + '$' + className.substring(lastDotIndex + 1); |
||||||
|
String innerClassResourcePath = ResourceLoader.CLASSPATH_URL_PREFIX + |
||||||
|
ClassUtils.convertClassNameToResourcePath(innerClassName) + ClassUtils.CLASS_FILE_SUFFIX; |
||||||
|
Resource innerClassResource = this.resourceLoader.getResource(innerClassResourcePath); |
||||||
|
if (innerClassResource.exists()) { |
||||||
|
return getMetadataReader(innerClassResource); |
||||||
|
} |
||||||
|
} |
||||||
|
throw ex; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
@ -1,78 +0,0 @@ |
|||||||
/* |
|
||||||
* Copyright 2002-present 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.core.type; |
|
||||||
|
|
||||||
import java.net.URL; |
|
||||||
|
|
||||||
import org.jspecify.annotations.Nullable; |
|
||||||
import org.junit.jupiter.api.Test; |
|
||||||
|
|
||||||
import org.springframework.core.io.Resource; |
|
||||||
import org.springframework.core.io.UrlResource; |
|
||||||
import org.springframework.core.testfixture.EnabledForTestGroups; |
|
||||||
import org.springframework.core.type.classreading.CachingMetadataReaderFactory; |
|
||||||
import org.springframework.core.type.classreading.MetadataReader; |
|
||||||
import org.springframework.core.type.classreading.MetadataReaderFactory; |
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.assertThat; |
|
||||||
import static org.springframework.core.testfixture.TestGroup.LONG_RUNNING; |
|
||||||
|
|
||||||
/** |
|
||||||
* Tests for checking the behaviour of {@link CachingMetadataReaderFactory} under |
|
||||||
* load. If the cache is not controlled, this test should fail with an out of memory |
|
||||||
* exception around entry 5k. |
|
||||||
* |
|
||||||
* @author Costin Leau |
|
||||||
* @author Sam Brannen |
|
||||||
*/ |
|
||||||
@EnabledForTestGroups(LONG_RUNNING) |
|
||||||
class CachingMetadataReaderLeakTests { |
|
||||||
|
|
||||||
private static final int ITEMS_TO_LOAD = 9999; |
|
||||||
|
|
||||||
private final MetadataReaderFactory mrf = new CachingMetadataReaderFactory(); |
|
||||||
|
|
||||||
@Test |
|
||||||
void significantLoad() throws Exception { |
|
||||||
// the biggest public class in the JDK (>60k)
|
|
||||||
URL url = getClass().getResource("/java/awt/Component.class"); |
|
||||||
assertThat(url).isNotNull(); |
|
||||||
|
|
||||||
// look at a LOT of items
|
|
||||||
for (int i = 0; i < ITEMS_TO_LOAD; i++) { |
|
||||||
Resource resource = new UrlResource(url) { |
|
||||||
|
|
||||||
@Override |
|
||||||
public boolean equals(@Nullable Object obj) { |
|
||||||
return (obj == this); |
|
||||||
} |
|
||||||
|
|
||||||
@Override |
|
||||||
public int hashCode() { |
|
||||||
return System.identityHashCode(this); |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
MetadataReader reader = mrf.getMetadataReader(resource); |
|
||||||
assertThat(reader).isNotNull(); |
|
||||||
} |
|
||||||
|
|
||||||
// useful for profiling to take snapshots
|
|
||||||
// System.in.read();
|
|
||||||
} |
|
||||||
|
|
||||||
} |
|
||||||
@ -0,0 +1,50 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2002-present 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.core.type.classreading; |
||||||
|
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test; |
||||||
|
|
||||||
|
import org.springframework.core.io.Resource; |
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.any; |
||||||
|
import static org.mockito.Mockito.mock; |
||||||
|
import static org.mockito.Mockito.times; |
||||||
|
import static org.mockito.Mockito.verify; |
||||||
|
import static org.mockito.Mockito.when; |
||||||
|
|
||||||
|
/** |
||||||
|
* Tests for {@link CachingMetadataReaderFactory}. |
||||||
|
*/ |
||||||
|
class CachingMetadataReaderFactoryTests { |
||||||
|
|
||||||
|
@Test |
||||||
|
void shouldCacheClassNameCalls() throws Exception { |
||||||
|
MetadataReaderFactory delegate = mock(MetadataReaderFactory.class); |
||||||
|
when(delegate.getMetadataReader(any(Resource.class))).thenReturn(mock(MetadataReader.class)); |
||||||
|
|
||||||
|
CachingMetadataReaderFactory readerFactory = new CachingMetadataReaderFactory(delegate); |
||||||
|
MetadataReader metadataReader = readerFactory.getMetadataReader(TestClass.class.getName()); |
||||||
|
metadataReader = readerFactory.getMetadataReader(TestClass.class.getName()); |
||||||
|
|
||||||
|
verify(delegate, times(1)).getMetadataReader(any(Resource.class)); |
||||||
|
} |
||||||
|
|
||||||
|
public static class TestClass { |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
Loading…
Reference in new issue