Skip to content

Commit 3c00156

Browse files
Add support for Wildcard Index
via the declarative WildcardIndexed annotation and the programatic WildcardIndex.
1 parent 24e909e commit 3c00156

File tree

9 files changed

+654
-19
lines changed

9 files changed

+654
-19
lines changed

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

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

118+
if(indexOptions.containsKey("wildcardProjection")) {
119+
ops.wildcardProjection(indexOptions.get("wildcardProjection", Document.class));
120+
}
121+
118122
return ops;
119123
};
120124
}

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

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

3131
enum Type {
32-
GEO, TEXT, DEFAULT, HASH;
32+
GEO, TEXT, DEFAULT, HASH, WILDCARD;
3333
}
3434

3535
private final String key;
@@ -48,7 +48,7 @@ private IndexField(String key, @Nullable Direction direction, @Nullable Type typ
4848
if (Type.GEO.equals(type) || Type.TEXT.equals(type)) {
4949
Assert.isNull(direction, "Geo/Text indexes must not have a direction!");
5050
} else {
51-
if (!Type.HASH.equals(type)) {
51+
if (!(Type.HASH.equals(type) || Type.WILDCARD.equals(type))) {
5252
Assert.notNull(direction, "Default indexes require a direction");
5353
}
5454
}
@@ -77,6 +77,17 @@ static IndexField hashed(String key) {
7777
return new IndexField(key, null, Type.HASH);
7878
}
7979

80+
/**
81+
* Creates a {@literal wildcard} {@link IndexField} for the given key.
82+
*
83+
* @param key must not be {@literal null} or empty.
84+
* @return new instance of {@link IndexField}.
85+
* @since 3.3
86+
*/
87+
static IndexField wildcard(String key) {
88+
return new IndexField(key, null, Type.WILDCARD);
89+
}
90+
8091
/**
8192
* Creates a geo {@link IndexField} for the given key.
8293
*
@@ -142,6 +153,16 @@ public boolean isHashed() {
142153
return Type.HASH.equals(type);
143154
}
144155

156+
/**
157+
* Returns whether the {@link IndexField} is contains a {@literal wildcard} expression.
158+
*
159+
* @return {@literal true} if {@link IndexField} contains a wildcard {@literal $**}.
160+
* @since 3.3
161+
*/
162+
public boolean isWildcard() {
163+
return Type.WILDCARD.equals(type);
164+
}
165+
145166
/*
146167
* (non-Javadoc)
147168
* @see java.lang.Object#equals(java.lang.Object)

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

+26
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@ public class IndexInfo {
5555
private @Nullable Duration expireAfter;
5656
private @Nullable String partialFilterExpression;
5757
private @Nullable Document collation;
58+
private @Nullable Document wildcardProjection;
5859

5960
public IndexInfo(List<IndexField> indexFields, String name, boolean unique, boolean sparse, String language) {
6061

@@ -99,6 +100,8 @@ public static IndexInfo indexInfoOf(Document sourceDocument) {
99100

100101
if (ObjectUtils.nullSafeEquals("hashed", value)) {
101102
indexFields.add(IndexField.hashed(key));
103+
} else if (key.contains("$**")) {
104+
indexFields.add(IndexField.wildcard(key));
102105
} else {
103106

104107
Double keyValue = new Double(value.toString());
@@ -131,6 +134,10 @@ public static IndexInfo indexInfoOf(Document sourceDocument) {
131134
info.expireAfter = Duration.ofSeconds(NumberUtils.convertNumberToTargetClass(expireAfterSeconds, Long.class));
132135
}
133136

137+
if (sourceDocument.containsKey("wildcardProjection")) {
138+
info.wildcardProjection = sourceDocument.get("wildcardProjection", Document.class);
139+
}
140+
134141
return info;
135142
}
136143

@@ -216,6 +223,16 @@ public Optional<Document> getCollation() {
216223
return Optional.ofNullable(collation);
217224
}
218225

226+
/**
227+
* Get {@literal wildcardProjection} information.
228+
*
229+
* @return {@link Optional#empty() empty} if not set.
230+
* @since 3.3
231+
*/
232+
public Optional<Document> getWildcardProjection() {
233+
return Optional.ofNullable(wildcardProjection);
234+
}
235+
219236
/**
220237
* Get the duration after which documents within the index expire.
221238
*
@@ -234,6 +251,14 @@ public boolean isHashed() {
234251
return getIndexFields().stream().anyMatch(IndexField::isHashed);
235252
}
236253

254+
/**
255+
* @return {@literal true} if a wildcard index field is present.
256+
* @since 3.3
257+
*/
258+
public boolean isWildcard() {
259+
return getIndexFields().stream().anyMatch(IndexField::isWildcard);
260+
}
261+
237262
@Override
238263
public String toString() {
239264

@@ -303,4 +328,5 @@ public boolean equals(Object obj) {
303328
}
304329
return true;
305330
}
331+
306332
}

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

+91-15
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
import org.springframework.data.mongodb.core.mapping.Document;
4747
import org.springframework.data.mongodb.core.mapping.MongoPersistentEntity;
4848
import org.springframework.data.mongodb.core.mapping.MongoPersistentProperty;
49+
import org.springframework.data.mongodb.core.query.Collation;
4950
import org.springframework.data.mongodb.util.BsonUtils;
5051
import org.springframework.data.mongodb.util.DotPath;
5152
import org.springframework.data.spel.EvaluationContextProvider;
@@ -121,6 +122,7 @@ public List<IndexDefinitionHolder> resolveIndexForEntity(MongoPersistentEntity<?
121122
List<IndexDefinitionHolder> indexInformation = new ArrayList<>();
122123
String collection = root.getCollection();
123124
indexInformation.addAll(potentiallyCreateCompoundIndexDefinitions("", collection, root));
125+
indexInformation.addAll(potentiallyCreateWildcardIndexDefinitions("", collection, root));
124126
indexInformation.addAll(potentiallyCreateTextIndexDefinition(root, collection));
125127

126128
root.doWithProperties((PropertyHandler<MongoPersistentProperty>) property -> this
@@ -162,17 +164,18 @@ private void potentiallyAddIndexForProperty(MongoPersistentEntity<?> root, Mongo
162164
* @return List of {@link IndexDefinitionHolder} representing indexes for given type and its referenced property
163165
* types. Will never be {@code null}.
164166
*/
165-
private List<IndexDefinitionHolder> resolveIndexForClass( TypeInformation<?> type, String dotPath,
166-
Path path, String collection, CycleGuard guard) {
167+
private List<IndexDefinitionHolder> resolveIndexForClass(TypeInformation<?> type, String dotPath, Path path,
168+
String collection, CycleGuard guard) {
167169

168170
return resolveIndexForEntity(mappingContext.getRequiredPersistentEntity(type), dotPath, path, collection, guard);
169171
}
170172

171-
private List<IndexDefinitionHolder> resolveIndexForEntity(MongoPersistentEntity<?> entity, String dotPath,
172-
Path path, String collection, CycleGuard guard) {
173+
private List<IndexDefinitionHolder> resolveIndexForEntity(MongoPersistentEntity<?> entity, String dotPath, Path path,
174+
String collection, CycleGuard guard) {
173175

174176
List<IndexDefinitionHolder> indexInformation = new ArrayList<>();
175177
indexInformation.addAll(potentiallyCreateCompoundIndexDefinitions(dotPath, collection, entity));
178+
indexInformation.addAll(potentiallyCreateWildcardIndexDefinitions(dotPath, collection, entity));
176179

177180
entity.doWithProperties((PropertyHandler<MongoPersistentProperty>) property -> this
178181
.guardAndPotentiallyAddIndexForProperty(property, dotPath, path, collection, indexInformation, guard));
@@ -196,15 +199,15 @@ private void guardAndPotentiallyAddIndexForProperty(MongoPersistentProperty pers
196199

197200
if (persistentProperty.isEntity()) {
198201
try {
199-
indexes.addAll(resolveIndexForEntity(mappingContext.getPersistentEntity(persistentProperty), propertyDotPath.toString(),
200-
propertyPath, collection, guard));
202+
indexes.addAll(resolveIndexForEntity(mappingContext.getPersistentEntity(persistentProperty),
203+
propertyDotPath.toString(), propertyPath, collection, guard));
201204
} catch (CyclicPropertyReferenceException e) {
202205
LOGGER.info(e.getMessage());
203206
}
204207
}
205208

206-
List<IndexDefinitionHolder> indexDefinitions = createIndexDefinitionHolderForProperty(propertyDotPath.toString(), collection,
207-
persistentProperty);
209+
List<IndexDefinitionHolder> indexDefinitions = createIndexDefinitionHolderForProperty(propertyDotPath.toString(),
210+
collection, persistentProperty);
208211

209212
if (!indexDefinitions.isEmpty()) {
210213
indexes.addAll(indexDefinitions);
@@ -232,6 +235,11 @@ private List<IndexDefinitionHolder> createIndexDefinitionHolderForProperty(Strin
232235
if (persistentProperty.isAnnotationPresent(HashIndexed.class)) {
233236
indices.add(createHashedIndexDefinition(dotPath, collection, persistentProperty));
234237
}
238+
if (persistentProperty.isAnnotationPresent(WildcardIndexed.class)) {
239+
indices.add(createWildcardIndexDefinition(dotPath, collection,
240+
persistentProperty.getRequiredAnnotation(WildcardIndexed.class),
241+
mappingContext.getPersistentEntity(persistentProperty)));
242+
}
235243

236244
return indices;
237245
}
@@ -246,6 +254,18 @@ private List<IndexDefinitionHolder> potentiallyCreateCompoundIndexDefinitions(St
246254
return createCompoundIndexDefinitions(dotPath, collection, entity);
247255
}
248256

257+
private List<IndexDefinitionHolder> potentiallyCreateWildcardIndexDefinitions(String dotPath, String collection,
258+
MongoPersistentEntity<?> entity) {
259+
260+
if (entity.findAnnotation(WildcardIndexed.class) == null) {
261+
return Collections.emptyList();
262+
}
263+
264+
return Collections.singletonList(new IndexDefinitionHolder(dotPath,
265+
createWildcardIndexDefinition(dotPath, collection, entity.getRequiredAnnotation(WildcardIndexed.class), entity),
266+
collection));
267+
}
268+
249269
private Collection<? extends IndexDefinitionHolder> potentiallyCreateTextIndexDefinition(
250270
MongoPersistentEntity<?> root, String collection) {
251271

@@ -292,9 +312,8 @@ private Collection<? extends IndexDefinitionHolder> potentiallyCreateTextIndexDe
292312

293313
}
294314

295-
private void appendTextIndexInformation(DotPath dotPath, Path path,
296-
TextIndexDefinitionBuilder indexDefinitionBuilder, MongoPersistentEntity<?> entity,
297-
TextIndexIncludeOptions includeOptions, CycleGuard guard) {
315+
private void appendTextIndexInformation(DotPath dotPath, Path path, TextIndexDefinitionBuilder indexDefinitionBuilder,
316+
MongoPersistentEntity<?> entity, TextIndexIncludeOptions includeOptions, CycleGuard guard) {
298317

299318
entity.doWithProperties(new PropertyHandler<MongoPersistentProperty>() {
300319

@@ -311,8 +330,7 @@ public void doWithPersistentProperty(MongoPersistentProperty persistentProperty)
311330

312331
if (includeOptions.isForce() || indexed != null || persistentProperty.isEntity()) {
313332

314-
DotPath propertyDotPath = dotPath
315-
.append(persistentProperty.getFieldName());
333+
DotPath propertyDotPath = dotPath.append(persistentProperty.getFieldName());
316334

317335
Path propertyPath = path.append(persistentProperty);
318336

@@ -406,6 +424,32 @@ protected IndexDefinitionHolder createCompoundIndexDefinition(String dotPath, St
406424
return new IndexDefinitionHolder(dotPath, indexDefinition, collection);
407425
}
408426

427+
protected IndexDefinitionHolder createWildcardIndexDefinition(String dotPath, String collection,
428+
WildcardIndexed index, @Nullable MongoPersistentEntity<?> entity) {
429+
430+
WildcardIndex indexDefinition = new WildcardIndex(dotPath);
431+
432+
if (StringUtils.hasText(index.wildcardProjection())) {
433+
indexDefinition.wildcardProjection(evaluateWildcardProjection(index.wildcardProjection(), entity));
434+
}
435+
436+
if (!index.useGeneratedName()) {
437+
indexDefinition.named(pathAwareIndexName(index.name(), dotPath, entity, null));
438+
}
439+
440+
if (StringUtils.hasText(index.partialFilter())) {
441+
indexDefinition.partial(evaluatePartialFilter(index.partialFilter(), entity));
442+
}
443+
444+
if (StringUtils.hasText(index.collation())) {
445+
indexDefinition.collation(evaluateCollation(index.collation(), entity));
446+
} else if (entity != null && entity.hasCollation()) {
447+
indexDefinition.collation(entity.getCollation());
448+
}
449+
450+
return new IndexDefinitionHolder(dotPath, indexDefinition, collection);
451+
}
452+
409453
private org.bson.Document resolveCompoundIndexKeyFromStringDefinition(String dotPath, String keyDefinitionString,
410454
PersistentEntity<?, ?> entity) {
411455

@@ -510,6 +554,33 @@ private PartialIndexFilter evaluatePartialFilter(String filterExpression, Persis
510554
return PartialIndexFilter.of(BsonUtils.parse(filterExpression, null));
511555
}
512556

557+
private org.bson.Document evaluateWildcardProjection(String projectionExpression, PersistentEntity<?, ?> entity) {
558+
559+
Object result = evaluate(projectionExpression, getEvaluationContextForProperty(entity));
560+
561+
if (result instanceof org.bson.Document) {
562+
return (org.bson.Document) result;
563+
}
564+
565+
return BsonUtils.parse(projectionExpression, null);
566+
}
567+
568+
private Collation evaluateCollation(String collationExpression, PersistentEntity<?, ?> entity) {
569+
570+
Object result = evaluate(collationExpression, getEvaluationContextForProperty(entity));
571+
if (result instanceof org.bson.Document) {
572+
return Collation.from((org.bson.Document) result);
573+
}
574+
if (result instanceof Collation) {
575+
return (Collation) result;
576+
}
577+
if (result instanceof String) {
578+
return Collation.parse(result.toString());
579+
}
580+
throw new IllegalStateException("Cannot parse collation " + result);
581+
582+
}
583+
513584
/**
514585
* Creates {@link HashedIndex} wrapped in {@link IndexDefinitionHolder} out of {@link HashIndexed} for a given
515586
* {@link MongoPersistentProperty}.
@@ -657,8 +728,8 @@ private void resolveAndAddIndexesForAssociation(Association<MongoPersistentPrope
657728
propertyDotPath));
658729
}
659730

660-
List<IndexDefinitionHolder> indexDefinitions = createIndexDefinitionHolderForProperty(propertyDotPath.toString(), collection,
661-
property);
731+
List<IndexDefinitionHolder> indexDefinitions = createIndexDefinitionHolderForProperty(propertyDotPath.toString(),
732+
collection, property);
662733

663734
if (!indexDefinitions.isEmpty()) {
664735
indexes.addAll(indexDefinitions);
@@ -998,6 +1069,11 @@ public org.bson.Document getIndexKeys() {
9981069
public org.bson.Document getIndexOptions() {
9991070
return indexDefinition.getIndexOptions();
10001071
}
1072+
1073+
@Override
1074+
public String toString() {
1075+
return "IndexDefinitionHolder{" + "indexKeys=" + getIndexKeys() + '}';
1076+
}
10011077
}
10021078

10031079
/**

0 commit comments

Comments
 (0)