Skip to content

Commit 4e6b12f

Browse files
committed
DATACMNS-1498 - Improved attribute lookup on RepositoryConfigurationSource.
1 parent 0bf160e commit 4e6b12f

File tree

5 files changed

+102
-8
lines changed

5 files changed

+102
-8
lines changed

src/main/java/org/springframework/data/repository/config/AnnotationRepositoryConfigurationSource.java

+29-3
Original file line numberDiff line numberDiff line change
@@ -328,9 +328,35 @@ public boolean shouldConsiderNestedRepositories() {
328328
*/
329329
@Override
330330
public Optional<String> getAttribute(String name) {
331+
return getAttribute(name, String.class);
332+
}
331333

332-
String attribute = attributes.getString(name);
333-
return StringUtils.hasText(attribute) ? Optional.of(attribute) : Optional.empty();
334+
/*
335+
* (non-Javadoc)
336+
* @see org.springframework.data.repository.config.RepositoryConfigurationSource#getAttribute(java.lang.String, java.lang.Class)
337+
*/
338+
@Override
339+
public <T> Optional<T> getAttribute(String name, Class<T> type) {
340+
341+
if (!attributes.containsKey(name)) {
342+
throw new IllegalArgumentException(String.format("No attribute named %s found!", name));
343+
}
344+
345+
Object value = attributes.get(name);
346+
347+
if (value == null) {
348+
return Optional.empty();
349+
}
350+
351+
Assert.isInstanceOf(type, value,
352+
() -> String.format("Attribute value for %s is of type %s but was expected to be of type %s!", name,
353+
value.getClass(), type));
354+
355+
Object result = String.class.isInstance(value) //
356+
? StringUtils.hasText((String) value) ? value : null //
357+
: value;
358+
359+
return Optional.ofNullable(type.cast(result));
334360
}
335361

336362
/*
@@ -342,7 +368,7 @@ public boolean usesExplicitFilters() {
342368
return hasExplicitFilters;
343369
}
344370

345-
/*
371+
/*
346372
* (non-Javadoc)
347373
* @see org.springframework.data.repository.config.RepositoryConfigurationSource#getBootstrapMode()
348374
*/

src/main/java/org/springframework/data/repository/config/RepositoryConfigurationSource.java

+28-2
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
import org.springframework.data.repository.query.QueryLookupStrategy;
2525
import org.springframework.data.util.Streamable;
2626
import org.springframework.lang.Nullable;
27+
import org.springframework.util.Assert;
2728

2829
/**
2930
* Interface containing the configurable options for the Spring Data repository subsystem.
@@ -103,6 +104,31 @@ public interface RepositoryConfigurationSource {
103104
*/
104105
Optional<String> getAttribute(String name);
105106

107+
/**
108+
* Returns the value for the attribute with the given name and type. The name is expected to be handed in camel-case.
109+
*
110+
* @param name must not be {@literal null} or empty.
111+
* @param type the type of the attribute to look up.
112+
* @return the attribute with the given name or {@link Optional#empty()} if not configured or empty.
113+
* @since 2.2
114+
*/
115+
<T> Optional<T> getAttribute(String name, Class<T> type);
116+
117+
/**
118+
* Returns the attribute value for the attribute of the given name.
119+
*
120+
* @param name must not be {@literal null} or empty.
121+
* @return the attribute with the given name and type.
122+
* @since 2.2
123+
*/
124+
default <T> T getRequiredAttribute(String name, Class<T> type) {
125+
126+
Assert.hasText(name, "Attribute name must not be null or empty!");
127+
128+
return getAttribute(name, type)
129+
.orElseThrow(() -> new IllegalArgumentException(String.format("No attribute named %s found!", name)));
130+
}
131+
106132
/**
107133
* Returns whether the configuration uses explicit filtering to scan for repository types.
108134
*
@@ -131,7 +157,7 @@ public interface RepositoryConfigurationSource {
131157
/**
132158
* Returns the {@link ImplementationDetectionConfiguration} to be used to scan for custom implementations of the
133159
* repository instances to be created from this {@link RepositoryConfigurationSource}.
134-
*
160+
*
135161
* @param factory
136162
* @return will never be {@literal null}.
137163
* @since 2.1
@@ -140,7 +166,7 @@ public interface RepositoryConfigurationSource {
140166

141167
/**
142168
* Defines the repository {@link BootstrapMode} to be used.
143-
*
169+
*
144170
* @return
145171
* @since 2.1
146172
*/

src/main/java/org/springframework/data/repository/config/XmlRepositoryConfigurationSource.java

+14-2
Original file line numberDiff line numberDiff line change
@@ -188,7 +188,6 @@ private Optional<String> getNullDefaultedAttribute(Element element, String attri
188188
*/
189189
@Override
190190
public boolean shouldConsiderNestedRepositories() {
191-
192191
return getNullDefaultedAttribute(element, CONSIDER_NESTED_REPOSITORIES).map(Boolean::parseBoolean).orElse(false);
193192
}
194193

@@ -205,6 +204,19 @@ public Optional<String> getAttribute(String name) {
205204
return StringUtils.hasText(attribute) ? Optional.of(attribute) : Optional.empty();
206205
}
207206

207+
/*
208+
* (non-Javadoc)
209+
* @see org.springframework.data.repository.config.RepositoryConfigurationSource#getAttribute(java.lang.String, java.lang.Class)
210+
*/
211+
@SuppressWarnings("unchecked")
212+
@Override
213+
public <T> Optional<T> getAttribute(String name, Class<T> type) {
214+
215+
Assert.isAssignable(String.class, type, "Only String attribute lookups are allowed for XML namespaces!");
216+
217+
return (Optional<T>) getAttribute(name);
218+
}
219+
208220
/*
209221
* (non-Javadoc)
210222
* @see org.springframework.data.repository.config.RepositoryConfigurationSource#usesExplicitFilters()
@@ -214,7 +226,7 @@ public boolean usesExplicitFilters() {
214226
return !(this.includeFilters.isEmpty() && this.excludeFilters.isEmpty());
215227
}
216228

217-
/*
229+
/*
218230
* (non-Javadoc)
219231
* @see org.springframework.data.repository.config.RepositoryConfigurationSource#getBootstrapMode()
220232
*/

src/test/java/org/springframework/data/repository/config/AnnotationRepositoryConfigurationSourceUnitTests.java

+29
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@
3333
import org.springframework.core.io.ResourceLoader;
3434
import org.springframework.core.type.AnnotationMetadata;
3535
import org.springframework.core.type.StandardAnnotationMetadata;
36+
import org.springframework.data.repository.PagingAndSortingRepository;
3637
import org.springframework.data.util.Streamable;
3738

3839
/**
@@ -138,6 +139,34 @@ public void ignoresMissingRepositoryBaseClassNameAttribute() {
138139
assertThat(configurationSource.getRepositoryBaseClassName()).isNotPresent();
139140
}
140141

142+
@Test // DATACMNS-1498
143+
public void allowsLookupOfNonStringAttribute() {
144+
145+
RepositoryConfigurationSource source = getConfigSource(DefaultConfiguration.class);
146+
147+
assertThat(source.getAttribute("repositoryBaseClass", Class.class)).hasValue(PagingAndSortingRepository.class);
148+
assertThat(source.getRequiredAttribute("repositoryBaseClass", Class.class))
149+
.isEqualTo(PagingAndSortingRepository.class);
150+
}
151+
152+
@Test // DATACMNS-1498
153+
public void rejectsInvalidAttributeName() {
154+
155+
RepositoryConfigurationSource source = getConfigSource(DefaultConfiguration.class);
156+
157+
assertThatExceptionOfType(IllegalArgumentException.class).isThrownBy(() -> source.getAttribute("fooBar"));
158+
}
159+
160+
@Test // DATACMNS-1498
161+
public void lookupOfEmptyStringExposesAbsentValue() {
162+
163+
RepositoryConfigurationSource source = getConfigSource(DefaultConfiguration.class);
164+
165+
assertThat(source.getAttribute("namedQueriesLocation", String.class)).isEmpty();
166+
assertThatExceptionOfType(IllegalArgumentException.class)
167+
.isThrownBy(() -> source.getRequiredAttribute("namedQueriesLocation", String.class));
168+
}
169+
141170
private AnnotationRepositoryConfigurationSource getConfigSource(Class<?> type) {
142171

143172
AnnotationMetadata metadata = new StandardAnnotationMetadata(type, true);

src/test/java/org/springframework/data/repository/config/XmlRepositoryConfigurationSourceUnitTests.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
package org.springframework.data.repository.config;
1717

1818
import static org.assertj.core.api.Assertions.*;
19+
import static org.mockito.ArgumentMatchers.*;
1920
import static org.mockito.Mockito.*;
2021

2122
import org.junit.Test;
@@ -40,8 +41,8 @@ public void translatesCamelCaseAttributeNameIntoXmlEquivalent() {
4041

4142
RepositoryConfigurationSource source = mock(XmlRepositoryConfigurationSource.class);
4243
ReflectionTestUtils.setField(source, "element", element);
43-
when(source.getAttribute(anyString())).thenCallRealMethod();
4444

45+
when(source.getAttribute(anyString())).thenCallRealMethod();
4546
when(element.getAttribute("some-xml-attribute")).thenReturn("value");
4647

4748
assertThat(source.getAttribute("someXmlAttribute")).hasValue("value");

0 commit comments

Comments
 (0)