Skip to content

Commit 0766f2e

Browse files
mp911dechristophstrobl
authored andcommitted
Do not create persistent properties for Map and Collection-like entities.
These types are expected to behave like maps and collections and should not carry properties. The only exception are types implementing Streamable as Streamable can be used in domain types. Resolves: #3056 Original Pull Request: #3059
1 parent 5502fcf commit 0766f2e

File tree

2 files changed

+81
-16
lines changed

2 files changed

+81
-16
lines changed

src/main/java/org/springframework/data/mapping/context/AbstractMappingContext.java

+41-8
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,7 @@
6464
import org.springframework.data.mapping.model.SimpleTypeHolder;
6565
import org.springframework.data.spel.EvaluationContextProvider;
6666
import org.springframework.data.spel.ExtensionAwareEvaluationContextProvider;
67+
import org.springframework.data.util.CustomCollections;
6768
import org.springframework.data.util.KotlinReflectionUtils;
6869
import org.springframework.data.util.NullableWrapperConverters;
6970
import org.springframework.data.util.Optionals;
@@ -458,16 +459,18 @@ private E doAddPersistentEntity(TypeInformation<?> typeInformation) {
458459
persistentEntities.put(typeInformation, Optional.of(entity));
459460
}
460461

461-
PropertyDescriptor[] pds = BeanUtils.getPropertyDescriptors(type);
462-
Map<String, PropertyDescriptor> descriptors = new HashMap<>();
462+
if (shouldCreateProperties(userTypeInformation)) {
463+
PropertyDescriptor[] pds = BeanUtils.getPropertyDescriptors(type);
464+
Map<String, PropertyDescriptor> descriptors = new HashMap<>();
463465

464-
for (PropertyDescriptor descriptor : pds) {
465-
descriptors.put(descriptor.getName(), descriptor);
466-
}
466+
for (PropertyDescriptor descriptor : pds) {
467+
descriptors.put(descriptor.getName(), descriptor);
468+
}
467469

468-
PersistentPropertyCreator persistentPropertyCreator = new PersistentPropertyCreator(entity, descriptors);
469-
ReflectionUtils.doWithFields(type, persistentPropertyCreator, PersistentPropertyFilter.INSTANCE);
470-
persistentPropertyCreator.addPropertiesForRemainingDescriptors();
470+
PersistentPropertyCreator persistentPropertyCreator = new PersistentPropertyCreator(entity, descriptors);
471+
ReflectionUtils.doWithFields(type, persistentPropertyCreator, PersistentPropertyFilter.INSTANCE);
472+
persistentPropertyCreator.addPropertiesForRemainingDescriptors();
473+
}
471474

472475
entity.verify();
473476

@@ -518,6 +521,35 @@ public Collection<TypeInformation<?>> getManagedTypes() {
518521
*/
519522
protected abstract P createPersistentProperty(Property property, E owner, SimpleTypeHolder simpleTypeHolder);
520523

524+
/**
525+
* Whether to create the {@link PersistentProperty}s for the given {@link TypeInformation}.
526+
*
527+
* @param typeInformation must not be {@literal null}.
528+
* @return {@literal true} properties should be created, {@literal false} otherwise
529+
*/
530+
protected boolean shouldCreateProperties(TypeInformation<?> typeInformation) {
531+
532+
Class<?> type = typeInformation.getType();
533+
534+
return !typeInformation.isMap() && !isCollectionLike(type);
535+
}
536+
537+
/**
538+
* In contrast to TypeInformation, we do not consider types implementing Streamable collection-like as domain types
539+
* can implement that type.
540+
*
541+
* @param type must not be {@literal null}.
542+
* @return
543+
* @see TypeInformation#isCollectionLike()
544+
*/
545+
private static boolean isCollectionLike(Class<?> type) {
546+
547+
return type.isArray() //
548+
|| Iterable.class.equals(type) //
549+
|| Streamable.class.equals(type) //
550+
|| Collection.class.isAssignableFrom(type) || CustomCollections.isCollection(type);
551+
}
552+
521553
@Override
522554
public void afterPropertiesSet() {
523555
initialize();
@@ -575,6 +607,7 @@ private PersistentPropertyCreator(E entity, Map<String, PropertyDescriptor> desc
575607
this.remainingDescriptors = remainingDescriptors;
576608
}
577609

610+
@Override
578611
public void doWith(Field field) {
579612

580613
String fieldName = field.getName();

src/test/java/org/springframework/data/mapping/context/AbstractMappingContextUnitTests.java

+40-8
Original file line numberDiff line numberDiff line change
@@ -22,14 +22,7 @@
2222
import groovy.lang.MetaClass;
2323

2424
import java.time.LocalDateTime;
25-
import java.util.ArrayList;
26-
import java.util.Arrays;
27-
import java.util.Collection;
28-
import java.util.Collections;
29-
import java.util.List;
30-
import java.util.Map;
31-
import java.util.Optional;
32-
import java.util.TreeMap;
25+
import java.util.*;
3326
import java.util.function.Supplier;
3427

3528
import org.junit.jupiter.api.BeforeEach;
@@ -49,6 +42,7 @@
4942
import org.springframework.data.mapping.model.BasicPersistentEntity;
5043
import org.springframework.data.mapping.model.SimpleTypeHolder;
5144
import org.springframework.data.util.StreamUtils;
45+
import org.springframework.data.util.Streamable;
5246
import org.springframework.data.util.TypeInformation;
5347

5448
/**
@@ -285,6 +279,20 @@ void shouldNotCreatePersistentEntityForListButItsGenericTypeArgument() {
285279
.doesNotContain(List.class, ArrayList.class);
286280
}
287281

282+
@Test // GH-2390
283+
void doesNotCreatePropertiesForMapAndCollectionTypes() {
284+
285+
assertThat(context.getPersistentEntity(HashSet.class)).isEmpty();
286+
assertThat(context.getPersistentEntity(HashMap.class)).isEmpty();
287+
}
288+
289+
@Test // GH-2390
290+
void createsPropertiesForStreamableAndIterableTypes() {
291+
292+
assertThat(context.getPersistentEntity(MyStreamable.class)).hasSize(1);
293+
assertThat(context.getPersistentEntity(MyIterable.class)).hasSize(1);
294+
}
295+
288296
@Test // GH-2390
289297
void detectsEntityTypeEvenIfSimpleTypeHolderConsidersCollectionsSimple() {
290298

@@ -534,4 +542,28 @@ static abstract class Base$$SpringProxy$873fa2e extends Base implements SpringPr
534542

535543
}
536544

545+
static class MyIterable implements Iterable<StreamComponent> {
546+
547+
String name;
548+
549+
@Override
550+
public Iterator<StreamComponent> iterator() {
551+
return Collections.emptyIterator();
552+
}
553+
}
554+
555+
static class MyStreamable implements Streamable<StreamComponent> {
556+
557+
String name;
558+
559+
@Override
560+
public Iterator<StreamComponent> iterator() {
561+
return Collections.emptyIterator();
562+
}
563+
}
564+
565+
record StreamComponent(String name) {
566+
567+
}
568+
537569
}

0 commit comments

Comments
 (0)