Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Do not create persistent properties for Map and Collection-like entities #3059

Closed
wants to merge 3 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

<groupId>org.springframework.data</groupId>
<artifactId>spring-data-commons</artifactId>
<version>3.2.4-SNAPSHOT</version>
<version>3.2.4-GH-3056-SNAPSHOT</version>

<name>Spring Data Core</name>
<description>Core Spring concepts underpinning every Spring Data module.</description>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,7 @@
import org.springframework.data.mapping.model.SimpleTypeHolder;
import org.springframework.data.spel.EvaluationContextProvider;
import org.springframework.data.spel.ExtensionAwareEvaluationContextProvider;
import org.springframework.data.util.CustomCollections;
import org.springframework.data.util.KotlinReflectionUtils;
import org.springframework.data.util.NullableWrapperConverters;
import org.springframework.data.util.Optionals;
Expand Down Expand Up @@ -143,7 +144,6 @@ public void setApplicationContext(ApplicationContext applicationContext) throws
*
* @param initialEntitySet
* @see #setManagedTypes(ManagedTypes)
*
*/
public void setInitialEntitySet(Set<? extends Class<?>> initialEntitySet) {
setManagedTypes(ManagedTypes.fromIterable(initialEntitySet));
Expand Down Expand Up @@ -415,16 +415,18 @@ private E doAddPersistentEntity(TypeInformation<?> typeInformation) {
persistentEntities.put(typeInformation, Optional.of(entity));
}

PropertyDescriptor[] pds = BeanUtils.getPropertyDescriptors(type);
Map<String, PropertyDescriptor> descriptors = new HashMap<>();
if (shouldCreateProperties(userTypeInformation)) {
PropertyDescriptor[] pds = BeanUtils.getPropertyDescriptors(type);
Map<String, PropertyDescriptor> descriptors = new HashMap<>();

for (PropertyDescriptor descriptor : pds) {
descriptors.put(descriptor.getName(), descriptor);
}
for (PropertyDescriptor descriptor : pds) {
descriptors.put(descriptor.getName(), descriptor);
}

PersistentPropertyCreator persistentPropertyCreator = new PersistentPropertyCreator(entity, descriptors);
ReflectionUtils.doWithFields(type, persistentPropertyCreator, PersistentPropertyFilter.INSTANCE);
persistentPropertyCreator.addPropertiesForRemainingDescriptors();
PersistentPropertyCreator persistentPropertyCreator = new PersistentPropertyCreator(entity, descriptors);
ReflectionUtils.doWithFields(type, persistentPropertyCreator, PersistentPropertyFilter.INSTANCE);
persistentPropertyCreator.addPropertiesForRemainingDescriptors();
}

entity.verify();

Expand Down Expand Up @@ -475,6 +477,35 @@ public Collection<TypeInformation<?>> getManagedTypes() {
*/
protected abstract P createPersistentProperty(Property property, E owner, SimpleTypeHolder simpleTypeHolder);

/**
* Whether to create the {@link PersistentProperty}s for the given {@link TypeInformation}.
*
* @param typeInformation must not be {@literal null}.
* @return {@literal true} properties should be created, {@literal false} otherwise
*/
protected boolean shouldCreateProperties(TypeInformation<?> typeInformation) {

Class<?> type = typeInformation.getType();

return !typeInformation.isMap() && !isCollectionLike(type);
}

/**
* In contrast to TypeInformation, we do not consider types implementing Streamable collection-like as domain types
* can implement that type.
*
* @param type must not be {@literal null}.
* @return
* @see TypeInformation#isCollectionLike()
*/
private static boolean isCollectionLike(Class<?> type) {

return type.isArray() //
|| Iterable.class.equals(type) //
|| Streamable.class.equals(type) //
|| Collection.class.isAssignableFrom(type) || CustomCollections.isCollection(type);
}

@Override
public void afterPropertiesSet() {
initialize();
Expand Down Expand Up @@ -532,6 +563,7 @@ private PersistentPropertyCreator(E entity, Map<String, PropertyDescriptor> desc
this.remainingDescriptors = remainingDescriptors;
}

@Override
public void doWith(Field field) {

String fieldName = field.getName();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -157,15 +157,6 @@ public TypeInformation<?> getTypeInformation() {

@Override
public Iterable<? extends TypeInformation<?>> getPersistentEntityTypeInformation() {

if (isMap() || isCollectionLike()) {
return entityTypeInformation.get();
}

if (!isEntity()) {
return Collections.emptySet();
}

return entityTypeInformation.get();
}

Expand Down Expand Up @@ -292,6 +283,7 @@ public Class<?> getActualType() {
return getActualTypeInformation().getType();
}

@Override
public boolean usePropertyAccess() {
return usePropertyAccess.get();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,14 +22,7 @@
import groovy.lang.MetaClass;

import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.TreeMap;
import java.util.*;
import java.util.function.Supplier;

import org.junit.jupiter.api.BeforeEach;
Expand All @@ -49,6 +42,7 @@
import org.springframework.data.mapping.model.BasicPersistentEntity;
import org.springframework.data.mapping.model.SimpleTypeHolder;
import org.springframework.data.util.StreamUtils;
import org.springframework.data.util.Streamable;
import org.springframework.data.util.TypeInformation;

/**
Expand Down Expand Up @@ -285,6 +279,20 @@ void shouldNotCreatePersistentEntityForListButItsGenericTypeArgument() {
.doesNotContain(List.class, ArrayList.class);
}

@Test // GH-2390
void doesNotCreatePropertiesForMapAndCollectionTypes() {

assertThat(context.getPersistentEntity(HashSet.class)).isEmpty();
assertThat(context.getPersistentEntity(HashMap.class)).isEmpty();
}

@Test // GH-2390
void createsPropertiesForStreamableAndIterableTypes() {

assertThat(context.getPersistentEntity(MyStreamable.class)).hasSize(1);
assertThat(context.getPersistentEntity(MyIterable.class)).hasSize(1);
}

@Test // GH-2390
void detectsEntityTypeEvenIfSimpleTypeHolderConsidersCollectionsSimple() {

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

}

static class MyIterable implements Iterable<StreamComponent> {

String name;

@Override
public Iterator<StreamComponent> iterator() {
return Collections.emptyIterator();
}
}

static class MyStreamable implements Streamable<StreamComponent> {

String name;

@Override
public Iterator<StreamComponent> iterator() {
return Collections.emptyIterator();
}
}

record StreamComponent(String name) {

}

}
Loading