Compare commits
3 Commits
labs/antor
...
issue/4426
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
53305065c3 | ||
|
|
3b5ec24332 | ||
|
|
a473cb4322 |
2
pom.xml
2
pom.xml
@@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
<groupId>org.springframework.data</groupId>
|
<groupId>org.springframework.data</groupId>
|
||||||
<artifactId>spring-data-mongodb-parent</artifactId>
|
<artifactId>spring-data-mongodb-parent</artifactId>
|
||||||
<version>4.2.0-SNAPSHOT</version>
|
<version>4.2.x-4426-SNAPSHOT</version>
|
||||||
<packaging>pom</packaging>
|
<packaging>pom</packaging>
|
||||||
|
|
||||||
<name>Spring Data MongoDB</name>
|
<name>Spring Data MongoDB</name>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.data</groupId>
|
<groupId>org.springframework.data</groupId>
|
||||||
<artifactId>spring-data-mongodb-parent</artifactId>
|
<artifactId>spring-data-mongodb-parent</artifactId>
|
||||||
<version>4.2.0-SNAPSHOT</version>
|
<version>4.2.x-4426-SNAPSHOT</version>
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.data</groupId>
|
<groupId>org.springframework.data</groupId>
|
||||||
<artifactId>spring-data-mongodb-parent</artifactId>
|
<artifactId>spring-data-mongodb-parent</artifactId>
|
||||||
<version>4.2.0-SNAPSHOT</version>
|
<version>4.2.x-4426-SNAPSHOT</version>
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
<parent>
|
<parent>
|
||||||
<groupId>org.springframework.data</groupId>
|
<groupId>org.springframework.data</groupId>
|
||||||
<artifactId>spring-data-mongodb-parent</artifactId>
|
<artifactId>spring-data-mongodb-parent</artifactId>
|
||||||
<version>4.2.0-SNAPSHOT</version>
|
<version>4.2.x-4426-SNAPSHOT</version>
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|||||||
@@ -1089,7 +1089,6 @@ public class QueryMapper {
|
|||||||
protected static class MetadataBackedField extends Field {
|
protected static class MetadataBackedField extends Field {
|
||||||
|
|
||||||
private static final Pattern POSITIONAL_PARAMETER_PATTERN = Pattern.compile("\\.\\$(\\[.*?\\])?");
|
private static final Pattern POSITIONAL_PARAMETER_PATTERN = Pattern.compile("\\.\\$(\\[.*?\\])?");
|
||||||
private static final Pattern DOT_POSITIONAL_PATTERN = Pattern.compile("\\.\\d+(?!$)");
|
|
||||||
private static final String INVALID_ASSOCIATION_REFERENCE = "Invalid path reference %s; Associations can only be pointed to directly or via their id property";
|
private static final String INVALID_ASSOCIATION_REFERENCE = "Invalid path reference %s; Associations can only be pointed to directly or via their id property";
|
||||||
|
|
||||||
private final MongoPersistentEntity<?> entity;
|
private final MongoPersistentEntity<?> entity;
|
||||||
@@ -1231,14 +1230,13 @@ public class QueryMapper {
|
|||||||
private PersistentPropertyPath<MongoPersistentProperty> getPath(String pathExpression,
|
private PersistentPropertyPath<MongoPersistentProperty> getPath(String pathExpression,
|
||||||
@Nullable MongoPersistentProperty sourceProperty) {
|
@Nullable MongoPersistentProperty sourceProperty) {
|
||||||
|
|
||||||
String rawPath = removePlaceholders(POSITIONAL_OPERATOR,
|
|
||||||
removePlaceholders(DOT_POSITIONAL_PATTERN, pathExpression));
|
|
||||||
|
|
||||||
if (sourceProperty != null && sourceProperty.getOwner().equals(entity)) {
|
if (sourceProperty != null && sourceProperty.getOwner().equals(entity)) {
|
||||||
return mappingContext.getPersistentPropertyPath(
|
return mappingContext.getPersistentPropertyPath(
|
||||||
PropertyPath.from(Pattern.quote(sourceProperty.getName()), entity.getTypeInformation()));
|
PropertyPath.from(Pattern.quote(sourceProperty.getName()), entity.getTypeInformation()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String rawPath = resolvePath(pathExpression);
|
||||||
|
|
||||||
PropertyPath path = forName(rawPath);
|
PropertyPath path = forName(rawPath);
|
||||||
if (path == null || isPathToJavaLangClassProperty(path)) {
|
if (path == null || isPathToJavaLangClassProperty(path)) {
|
||||||
return null;
|
return null;
|
||||||
@@ -1333,6 +1331,36 @@ public class QueryMapper {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static String resolvePath(String source) {
|
||||||
|
|
||||||
|
String[] segments = source.split("\\.");
|
||||||
|
if (segments.length == 1) {
|
||||||
|
return source;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<String> path = new ArrayList<>();
|
||||||
|
|
||||||
|
/* always start from a property, so we can skip the first segment.
|
||||||
|
from there remove any position placeholder */
|
||||||
|
for (String segment : Arrays.copyOfRange(segments, 1, segments.length)) {
|
||||||
|
if (segment.startsWith("[") && segment.endsWith("]")) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (segment.matches("\\d+")) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
path.add(segment);
|
||||||
|
}
|
||||||
|
|
||||||
|
// when property is followed only by placeholders eg. 'values.0.3.90'
|
||||||
|
if (path.isEmpty()) {
|
||||||
|
return source;
|
||||||
|
}
|
||||||
|
|
||||||
|
path.add(0, segments[0]);
|
||||||
|
return StringUtils.collectionToDelimitedString(path, ".");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the {@link Converter} to be used to created the mapped key. Default implementation will use
|
* Return the {@link Converter} to be used to created the mapped key. Default implementation will use
|
||||||
* {@link PropertyToFieldNameConverter}.
|
* {@link PropertyToFieldNameConverter}.
|
||||||
|
|||||||
@@ -31,6 +31,8 @@ import org.bson.types.ObjectId;
|
|||||||
import org.junit.jupiter.api.BeforeEach;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.api.extension.ExtendWith;
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
|
import org.junit.jupiter.params.provider.ValueSource;
|
||||||
import org.mockito.Mockito;
|
import org.mockito.Mockito;
|
||||||
import org.mockito.junit.jupiter.MockitoExtension;
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
import org.springframework.core.convert.converter.Converter;
|
import org.springframework.core.convert.converter.Converter;
|
||||||
@@ -1207,24 +1209,26 @@ class UpdateMapperUnitTests {
|
|||||||
assertThat(mappedUpdate).isEqualTo(new org.bson.Document("$set", new org.bson.Document("levelOne.a.b.d", "e")));
|
assertThat(mappedUpdate).isEqualTo(new org.bson.Document("$set", new org.bson.Document("levelOne.a.b.d", "e")));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test // GH-3775
|
@ParameterizedTest // GH-3775, GH-4426
|
||||||
void mapNestedIntegerFieldCorrectly() {
|
@ValueSource(strings = {"levelOne.0.1.3", "levelOne.0.1.32", "levelOne2.0.1.32", "levelOne2.0.1.320"})
|
||||||
|
void mapNestedIntegerFieldCorrectly(String path) {
|
||||||
|
|
||||||
Update update = new Update().set("levelOne.0.1.3", "4");
|
Update update = new Update().set(path, "4");
|
||||||
Document mappedUpdate = mapper.getMappedObject(update.getUpdateObject(),
|
Document mappedUpdate = mapper.getMappedObject(update.getUpdateObject(),
|
||||||
context.getPersistentEntity(EntityWithNestedMap.class));
|
context.getPersistentEntity(EntityWithNestedMap.class));
|
||||||
|
|
||||||
assertThat(mappedUpdate).isEqualTo(new org.bson.Document("$set", new org.bson.Document("levelOne.0.1.3", "4")));
|
assertThat(mappedUpdate).isEqualTo(new org.bson.Document("$set", new org.bson.Document(path, "4")));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test // GH-3775
|
@ParameterizedTest // GH-3775, GH-4426
|
||||||
void mapNestedMixedStringIntegerFieldCorrectly() {
|
@ValueSource(strings = {"levelOne.0.1.c", "levelOne.0.1.c.32", "levelOne2.0.1.32.c", "levelOne2.0.1.c.320"})
|
||||||
|
void mapNestedMixedStringIntegerFieldCorrectly(String path) {
|
||||||
|
|
||||||
Update update = new Update().set("levelOne.0.1.c", "4");
|
Update update = new Update().set(path, "4");
|
||||||
Document mappedUpdate = mapper.getMappedObject(update.getUpdateObject(),
|
Document mappedUpdate = mapper.getMappedObject(update.getUpdateObject(),
|
||||||
context.getPersistentEntity(EntityWithNestedMap.class));
|
context.getPersistentEntity(EntityWithNestedMap.class));
|
||||||
|
|
||||||
assertThat(mappedUpdate).isEqualTo(new org.bson.Document("$set", new org.bson.Document("levelOne.0.1.c", "4")));
|
assertThat(mappedUpdate).isEqualTo(new org.bson.Document("$set", new org.bson.Document(path, "4")));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test // GH-3775
|
@Test // GH-3775
|
||||||
@@ -1732,6 +1736,7 @@ class UpdateMapperUnitTests {
|
|||||||
|
|
||||||
static class EntityWithNestedMap {
|
static class EntityWithNestedMap {
|
||||||
Map<String, Map<String, Map<String, Object>>> levelOne;
|
Map<String, Map<String, Map<String, Object>>> levelOne;
|
||||||
|
Map<String, Map<String, Map<String, Object>>> levelOne2;
|
||||||
}
|
}
|
||||||
|
|
||||||
static class Customer {
|
static class Customer {
|
||||||
|
|||||||
Reference in New Issue
Block a user