46
46
import org .springframework .data .mongodb .core .mapping .Document ;
47
47
import org .springframework .data .mongodb .core .mapping .MongoPersistentEntity ;
48
48
import org .springframework .data .mongodb .core .mapping .MongoPersistentProperty ;
49
+ import org .springframework .data .mongodb .core .query .Collation ;
49
50
import org .springframework .data .mongodb .util .BsonUtils ;
50
51
import org .springframework .data .mongodb .util .DotPath ;
51
52
import org .springframework .data .spel .EvaluationContextProvider ;
@@ -121,6 +122,7 @@ public List<IndexDefinitionHolder> resolveIndexForEntity(MongoPersistentEntity<?
121
122
List <IndexDefinitionHolder > indexInformation = new ArrayList <>();
122
123
String collection = root .getCollection ();
123
124
indexInformation .addAll (potentiallyCreateCompoundIndexDefinitions ("" , collection , root ));
125
+ indexInformation .addAll (potentiallyCreateWildcardIndexDefinitions ("" , collection , root ));
124
126
indexInformation .addAll (potentiallyCreateTextIndexDefinition (root , collection ));
125
127
126
128
root .doWithProperties ((PropertyHandler <MongoPersistentProperty >) property -> this
@@ -162,17 +164,18 @@ private void potentiallyAddIndexForProperty(MongoPersistentEntity<?> root, Mongo
162
164
* @return List of {@link IndexDefinitionHolder} representing indexes for given type and its referenced property
163
165
* types. Will never be {@code null}.
164
166
*/
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 ) {
167
169
168
170
return resolveIndexForEntity (mappingContext .getRequiredPersistentEntity (type ), dotPath , path , collection , guard );
169
171
}
170
172
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 ) {
173
175
174
176
List <IndexDefinitionHolder > indexInformation = new ArrayList <>();
175
177
indexInformation .addAll (potentiallyCreateCompoundIndexDefinitions (dotPath , collection , entity ));
178
+ indexInformation .addAll (potentiallyCreateWildcardIndexDefinitions (dotPath , collection , entity ));
176
179
177
180
entity .doWithProperties ((PropertyHandler <MongoPersistentProperty >) property -> this
178
181
.guardAndPotentiallyAddIndexForProperty (property , dotPath , path , collection , indexInformation , guard ));
@@ -196,15 +199,15 @@ private void guardAndPotentiallyAddIndexForProperty(MongoPersistentProperty pers
196
199
197
200
if (persistentProperty .isEntity ()) {
198
201
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 ));
201
204
} catch (CyclicPropertyReferenceException e ) {
202
205
LOGGER .info (e .getMessage ());
203
206
}
204
207
}
205
208
206
- List <IndexDefinitionHolder > indexDefinitions = createIndexDefinitionHolderForProperty (propertyDotPath .toString (), collection ,
207
- persistentProperty );
209
+ List <IndexDefinitionHolder > indexDefinitions = createIndexDefinitionHolderForProperty (propertyDotPath .toString (),
210
+ collection , persistentProperty );
208
211
209
212
if (!indexDefinitions .isEmpty ()) {
210
213
indexes .addAll (indexDefinitions );
@@ -232,6 +235,11 @@ private List<IndexDefinitionHolder> createIndexDefinitionHolderForProperty(Strin
232
235
if (persistentProperty .isAnnotationPresent (HashIndexed .class )) {
233
236
indices .add (createHashedIndexDefinition (dotPath , collection , persistentProperty ));
234
237
}
238
+ if (persistentProperty .isAnnotationPresent (WildcardIndexed .class )) {
239
+ indices .add (createWildcardIndexDefinition (dotPath , collection ,
240
+ persistentProperty .getRequiredAnnotation (WildcardIndexed .class ),
241
+ mappingContext .getPersistentEntity (persistentProperty )));
242
+ }
235
243
236
244
return indices ;
237
245
}
@@ -246,6 +254,18 @@ private List<IndexDefinitionHolder> potentiallyCreateCompoundIndexDefinitions(St
246
254
return createCompoundIndexDefinitions (dotPath , collection , entity );
247
255
}
248
256
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
+
249
269
private Collection <? extends IndexDefinitionHolder > potentiallyCreateTextIndexDefinition (
250
270
MongoPersistentEntity <?> root , String collection ) {
251
271
@@ -292,9 +312,8 @@ private Collection<? extends IndexDefinitionHolder> potentiallyCreateTextIndexDe
292
312
293
313
}
294
314
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 ) {
298
317
299
318
entity .doWithProperties (new PropertyHandler <MongoPersistentProperty >() {
300
319
@@ -311,8 +330,7 @@ public void doWithPersistentProperty(MongoPersistentProperty persistentProperty)
311
330
312
331
if (includeOptions .isForce () || indexed != null || persistentProperty .isEntity ()) {
313
332
314
- DotPath propertyDotPath = dotPath
315
- .append (persistentProperty .getFieldName ());
333
+ DotPath propertyDotPath = dotPath .append (persistentProperty .getFieldName ());
316
334
317
335
Path propertyPath = path .append (persistentProperty );
318
336
@@ -406,6 +424,32 @@ protected IndexDefinitionHolder createCompoundIndexDefinition(String dotPath, St
406
424
return new IndexDefinitionHolder (dotPath , indexDefinition , collection );
407
425
}
408
426
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
+
409
453
private org .bson .Document resolveCompoundIndexKeyFromStringDefinition (String dotPath , String keyDefinitionString ,
410
454
PersistentEntity <?, ?> entity ) {
411
455
@@ -510,6 +554,33 @@ private PartialIndexFilter evaluatePartialFilter(String filterExpression, Persis
510
554
return PartialIndexFilter .of (BsonUtils .parse (filterExpression , null ));
511
555
}
512
556
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
+
513
584
/**
514
585
* Creates {@link HashedIndex} wrapped in {@link IndexDefinitionHolder} out of {@link HashIndexed} for a given
515
586
* {@link MongoPersistentProperty}.
@@ -657,8 +728,8 @@ private void resolveAndAddIndexesForAssociation(Association<MongoPersistentPrope
657
728
propertyDotPath ));
658
729
}
659
730
660
- List <IndexDefinitionHolder > indexDefinitions = createIndexDefinitionHolderForProperty (propertyDotPath .toString (), collection ,
661
- property );
731
+ List <IndexDefinitionHolder > indexDefinitions = createIndexDefinitionHolderForProperty (propertyDotPath .toString (),
732
+ collection , property );
662
733
663
734
if (!indexDefinitions .isEmpty ()) {
664
735
indexes .addAll (indexDefinitions );
@@ -998,6 +1069,11 @@ public org.bson.Document getIndexKeys() {
998
1069
public org .bson .Document getIndexOptions () {
999
1070
return indexDefinition .getIndexOptions ();
1000
1071
}
1072
+
1073
+ @ Override
1074
+ public String toString () {
1075
+ return "IndexDefinitionHolder{" + "indexKeys=" + getIndexKeys () + '}' ;
1076
+ }
1001
1077
}
1002
1078
1003
1079
/**
0 commit comments