Skip to content

Commit f3b90c2

Browse files
committed
Polishing.
Reformat code. Tweak javadoc. Reject wildcard projection usage on properties with a MappingException. Omit wildcard projections when declared on document types that are used as subdocument. See #3225 Original pull request: #3671.
1 parent d57c5a9 commit f3b90c2

File tree

9 files changed

+87
-23
lines changed

9 files changed

+87
-23
lines changed

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/IndexConverters.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -115,7 +115,7 @@ private static Converter<IndexDefinition, IndexOptions> getIndexDefinitionIndexO
115115
ops = ops.collation(fromDocument(indexOptions.get("collation", Document.class)));
116116
}
117117

118-
if(indexOptions.containsKey("wildcardProjection")) {
118+
if (indexOptions.containsKey("wildcardProjection")) {
119119
ops.wildcardProjection(indexOptions.get("wildcardProjection", Document.class));
120120
}
121121

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/index/IndexField.java

+13-2
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,17 @@
2929
public final class IndexField {
3030

3131
enum Type {
32-
GEO, TEXT, DEFAULT, HASH, WILDCARD;
32+
GEO, TEXT, DEFAULT,
33+
34+
/**
35+
* @since 2.2
36+
*/
37+
HASH,
38+
39+
/**
40+
* @since 3.3
41+
*/
42+
WILDCARD;
3343
}
3444

3545
private final String key;
@@ -78,7 +88,8 @@ static IndexField hashed(String key) {
7888
}
7989

8090
/**
81-
* Creates a {@literal wildcard} {@link IndexField} for the given key.
91+
* Creates a {@literal wildcard} {@link IndexField} for the given key. The {@code key} must follow the
92+
* {@code fieldName.$**} notation.
8293
*
8394
* @param key must not be {@literal null} or empty.
8495
* @return new instance of {@link IndexField}.

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/index/IndexInfo.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -100,7 +100,7 @@ public static IndexInfo indexInfoOf(Document sourceDocument) {
100100

101101
if (ObjectUtils.nullSafeEquals("hashed", value)) {
102102
indexFields.add(IndexField.hashed(key));
103-
} else if (key.contains("$**")) {
103+
} else if (key.endsWith("$**")) {
104104
indexFields.add(IndexField.wildcard(key));
105105
} else {
106106

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/index/MongoPersistentEntityIndexResolver.java

+22-2
Original file line numberDiff line numberDiff line change
@@ -119,6 +119,8 @@ public List<IndexDefinitionHolder> resolveIndexForEntity(MongoPersistentEntity<?
119119
Assert.notNull(document, () -> String
120120
.format("Entity %s is not a collection root. Make sure to annotate it with @Document!", root.getName()));
121121

122+
verifyWildcardIndexedProjection(root);
123+
122124
List<IndexDefinitionHolder> indexInformation = new ArrayList<>();
123125
String collection = root.getCollection();
124126
indexInformation.addAll(potentiallyCreateCompoundIndexDefinitions("", collection, root));
@@ -133,6 +135,24 @@ public List<IndexDefinitionHolder> resolveIndexForEntity(MongoPersistentEntity<?
133135
return indexInformation;
134136
}
135137

138+
private void verifyWildcardIndexedProjection(MongoPersistentEntity<?> entity) {
139+
140+
entity.doWithAll(it -> {
141+
142+
if (it.isAnnotationPresent(WildcardIndexed.class)) {
143+
144+
WildcardIndexed indexed = it.getRequiredAnnotation(WildcardIndexed.class);
145+
146+
if (!ObjectUtils.isEmpty(indexed.wildcardProjection())) {
147+
148+
throw new MappingException(String.format(
149+
"WildcardIndexed.wildcardProjection cannot be used on nested paths. Offending property: %s.%s",
150+
entity.getName(), it.getName()));
151+
}
152+
}
153+
});
154+
}
155+
136156
private void potentiallyAddIndexForProperty(MongoPersistentEntity<?> root, MongoPersistentProperty persistentProperty,
137157
List<IndexDefinitionHolder> indexes, CycleGuard guard) {
138158

@@ -257,7 +277,7 @@ private List<IndexDefinitionHolder> potentiallyCreateCompoundIndexDefinitions(St
257277
private List<IndexDefinitionHolder> potentiallyCreateWildcardIndexDefinitions(String dotPath, String collection,
258278
MongoPersistentEntity<?> entity) {
259279

260-
if (entity.findAnnotation(WildcardIndexed.class) == null) {
280+
if (!entity.isAnnotationPresent(WildcardIndexed.class)) {
261281
return Collections.emptyList();
262282
}
263283

@@ -429,7 +449,7 @@ protected IndexDefinitionHolder createWildcardIndexDefinition(String dotPath, St
429449

430450
WildcardIndex indexDefinition = new WildcardIndex(dotPath);
431451

432-
if (StringUtils.hasText(index.wildcardProjection())) {
452+
if (StringUtils.hasText(index.wildcardProjection()) && ObjectUtils.isEmpty(dotPath)) {
433453
indexDefinition.wildcardProjection(evaluateWildcardProjection(index.wildcardProjection(), entity));
434454
}
435455

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/index/WildcardIndex.java

+5-5
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,7 @@
5151
public class WildcardIndex extends Index {
5252

5353
private @Nullable String fieldName;
54-
private Map<String, Object> wildcardProjection = new LinkedHashMap<>();
54+
private final Map<String, Object> wildcardProjection = new LinkedHashMap<>();
5555

5656
/**
5757
* Create a new instance of {@link WildcardIndex} using {@code $**}.
@@ -97,7 +97,7 @@ public WildcardIndex named(String name) {
9797
/**
9898
* Unique option is not supported.
9999
*
100-
* @throws UnsupportedOperationException
100+
* @throws UnsupportedOperationException not supported for wildcard indexes.
101101
*/
102102
@Override
103103
public Index unique() {
@@ -107,7 +107,7 @@ public Index unique() {
107107
/**
108108
* ttl option is not supported.
109109
*
110-
* @throws UnsupportedOperationException
110+
* @throws UnsupportedOperationException not supported for wildcard indexes.
111111
*/
112112
@Override
113113
public Index expire(long seconds) {
@@ -117,7 +117,7 @@ public Index expire(long seconds) {
117117
/**
118118
* ttl option is not supported.
119119
*
120-
* @throws UnsupportedOperationException
120+
* @throws UnsupportedOperationException not supported for wildcard indexes.
121121
*/
122122
@Override
123123
public Index expire(long value, TimeUnit timeUnit) {
@@ -127,7 +127,7 @@ public Index expire(long value, TimeUnit timeUnit) {
127127
/**
128128
* ttl option is not supported.
129129
*
130-
* @throws UnsupportedOperationException
130+
* @throws UnsupportedOperationException not supported for wildcard indexes.
131131
*/
132132
@Override
133133
public Index expire(Duration duration) {

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/index/WildcardIndexed.java

+7-7
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@
3838
*
3939
* db.product.createIndex({ "$**" : 1 } , {})
4040
* </pre>
41-
*
41+
*
4242
* {@literal wildcardProjection} can be used to specify keys to in-/exclude in the index.
4343
*
4444
* <pre class="code">
@@ -65,7 +65,7 @@
6565
* <pre class="code">
6666
* &#64;Document
6767
* public class User {
68-
*
68+
*
6969
* private &#64;Id String id;
7070
*
7171
* &#64;WildcardIndexed
@@ -89,9 +89,9 @@
8989
* expression}. <br />
9090
* <br />
9191
* The name will only be applied as is when defined on root level. For usage on nested or embedded structures the
92-
* provided name will be prefixed with the path leading to the entity. <br />
93-
*
94-
* @return
92+
* provided name will be prefixed with the path leading to the entity.
93+
*
94+
* @return empty by default.
9595
*/
9696
String name() default "";
9797

@@ -115,8 +115,8 @@
115115
/**
116116
* Explicitly specify sub fields to be in-/excluded as a {@link org.bson.Document#parse(String) prasable} String.
117117
* <br />
118-
* <strong>NOTE: </strong>Can only be done on root level documents.
119-
*
118+
* <strong>NOTE: </strong>Can only be applied on root level documents.
119+
*
120120
* @return empty by default.
121121
*/
122122
String wildcardProjection() default "";

spring-data-mongodb/src/test/java/org/springframework/data/mongodb/core/index/MongoPersistentEntityIndexResolverUnitTests.java

+32-1
Original file line numberDiff line numberDiff line change
@@ -32,10 +32,12 @@
3232
import org.junit.runner.RunWith;
3333
import org.junit.runners.Suite;
3434
import org.junit.runners.Suite.SuiteClasses;
35+
3536
import org.springframework.core.annotation.AliasFor;
3637
import org.springframework.dao.InvalidDataAccessApiUsageException;
3738
import org.springframework.data.annotation.Id;
3839
import org.springframework.data.geo.Point;
40+
import org.springframework.data.mapping.MappingException;
3941
import org.springframework.data.mongodb.core.DocumentTestUtils;
4042
import org.springframework.data.mongodb.core.index.MongoPersistentEntityIndexResolver.IndexDefinitionHolder;
4143
import org.springframework.data.mongodb.core.index.MongoPersistentEntityIndexResolverUnitTests.CompoundIndexResolutionTests;
@@ -1333,6 +1335,20 @@ public void resolvesWildcardOnRoot() {
13331335
assertThat(indices).hasSize(1);
13341336
assertThat(indices.get(0)).satisfies(it -> {
13351337
assertThat(it.getIndexKeys()).containsEntry("$**", 1);
1338+
assertThat(it.getIndexOptions()).isEmpty();
1339+
});
1340+
}
1341+
1342+
@Test // GH-3225
1343+
public void resolvesWildcardWithProjectionOnRoot() {
1344+
1345+
List<IndexDefinitionHolder> indices = prepareMappingContextAndResolveIndexForType(
1346+
WithWildCardIndexHavingProjectionOnEntity.class);
1347+
assertThat(indices).hasSize(1);
1348+
assertThat(indices.get(0)).satisfies(it -> {
1349+
assertThat(it.getIndexKeys()).containsEntry("$**", 1);
1350+
assertThat(it.getIndexOptions()).containsEntry("wildcardProjection",
1351+
org.bson.Document.parse("{'_id' : 1, 'value' : 0}"));
13361352
});
13371353
}
13381354

@@ -1365,6 +1381,15 @@ public void resolvesWildcardTypeOfNestedProperty() {
13651381
assertThat(indices).hasSize(1);
13661382
assertThat(indices.get(0)).satisfies(it -> {
13671383
assertThat(it.getIndexKeys()).containsEntry("value.$**", 1);
1384+
assertThat(it.getIndexOptions()).hasSize(1).containsKey("name");
1385+
});
1386+
}
1387+
1388+
@Test // GH-3225
1389+
public void rejectsWildcardProjectionOnNestedPaths() {
1390+
1391+
assertThatExceptionOfType(MappingException.class).isThrownBy(() -> {
1392+
prepareMappingContextAndResolveIndexForType(WildcardIndexedProjectionOnNestedPath.class);
13681393
});
13691394
}
13701395

@@ -1647,10 +1672,16 @@ class WithWildCardIndexOnProperty {
16471672

16481673
}
16491674

1675+
@Document
1676+
class WildcardIndexedProjectionOnNestedPath {
1677+
1678+
@WildcardIndexed(wildcardProjection = "{}") String foo;
1679+
}
1680+
16501681
@Document
16511682
class WithWildCardOnEntityOfNested {
16521683

1653-
WithWildCardIndexOnEntity value;
1684+
WithWildCardIndexHavingProjectionOnEntity value;
16541685

16551686
}
16561687

src/main/asciidoc/new-features.adoc

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
* Extended support for <<mapping-usage.document-references, referencing>> entities.
88
* Include/exclude `null` properties on write to `Document` through `@Field(write=…)`.
9+
* Support for <<mapping-usage-indexes.wildcard-index>>.
910

1011
[[new-features.3.2]]
1112
== What's New in Spring Data MongoDB 3.2

src/main/asciidoc/reference/mapping.adoc

+5-4
Original file line numberDiff line numberDiff line change
@@ -782,9 +782,9 @@ db.user.createIndex({ "userMetadata.$**" : 1 }, {})
782782
----
783783
====
784784

785-
The `@WildcardIndex` annotation allows a declarative index setup an can be added on either a type or property.
785+
The `@WildcardIndex` annotation allows a declarative index setup that can used either with a document type or property.
786786

787-
If placed on a type that is a root level domain entity (one having an `@Document` annotation) will advise the index creator to create a
787+
If placed on a type that is a root level domain entity (one annotated with `@Document`) , the index resolver will create a
788788
wildcard index for it.
789789

790790
.Wildcard index on domain type
@@ -794,7 +794,7 @@ wildcard index for it.
794794
@Document
795795
@WildcardIndexed
796796
public class Product {
797-
...
797+
// …
798798
}
799799
----
800800
[source,javascript]
@@ -828,7 +828,8 @@ db.user.createIndex(
828828
====
829829

830830
Wildcard indexes can also be expressed by adding the annotation directly to the field.
831-
Please note that `wildcardProjection` is not allowed on nested paths.
831+
Please note that `wildcardProjection` is not allowed on nested paths such as properties.
832+
Projections on types annotated with `@WildcardIndexed` are omitted during index creation.
832833

833834
.Wildcard index on property
834835
====

0 commit comments

Comments
 (0)