Skip to content

Commit f00991d

Browse files
committed
Polishing.
Rename Granularities/Granularity to Granularity and GranularityDefinition to proivide a more natural wording towards using predefined granularities. Validate presence of referenced properties through the TimeSeries annotation. Tweak Javadoc, reformat code, add unit tests. See #3731 Original pull request: #3732.
1 parent bacbd71 commit f00991d

File tree

11 files changed

+198
-112
lines changed

11 files changed

+198
-112
lines changed

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

+13-11
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,8 @@
2020
import org.springframework.data.mongodb.core.mapping.Field;
2121
import org.springframework.data.mongodb.core.query.Collation;
2222
import org.springframework.data.mongodb.core.schema.MongoJsonSchema;
23-
import org.springframework.data.mongodb.core.timeseries.Granularities;
2423
import org.springframework.data.mongodb.core.timeseries.Granularity;
24+
import org.springframework.data.mongodb.core.timeseries.GranularityDefinition;
2525
import org.springframework.data.mongodb.core.validation.Validator;
2626
import org.springframework.data.util.Optionals;
2727
import org.springframework.lang.Nullable;
@@ -100,7 +100,7 @@ public static CollectionOptions empty() {
100100
/**
101101
* Quick way to set up {@link CollectionOptions} for a Time Series collection. For more advanced settings use
102102
* {@link #timeSeries(TimeSeriesOptions)}.
103-
*
103+
*
104104
* @param timeField The name of the property which contains the date in each time series document. Must not be
105105
* {@literal null}.
106106
* @return new instance of {@link CollectionOptions}.
@@ -454,12 +454,13 @@ public static class TimeSeriesOptions {
454454

455455
private final String timeField;
456456

457-
@Nullable //
458-
private String metaField;
457+
private @Nullable final String metaField;
458+
459+
private final GranularityDefinition granularity;
459460

460-
private Granularity granularity;
461+
private TimeSeriesOptions(String timeField, @Nullable String metaField, GranularityDefinition granularity) {
461462

462-
private TimeSeriesOptions(String timeField, @Nullable String metaField, Granularity granularity) {
463+
Assert.hasText(timeField, "Time field must not be empty or null!");
463464

464465
this.timeField = timeField;
465466
this.metaField = metaField;
@@ -475,7 +476,7 @@ private TimeSeriesOptions(String timeField, @Nullable String metaField, Granular
475476
* @return new instance of {@link TimeSeriesOptions}.
476477
*/
477478
public static TimeSeriesOptions timeSeries(String timeField) {
478-
return new TimeSeriesOptions(timeField, null, Granularities.DEFAULT);
479+
return new TimeSeriesOptions(timeField, null, Granularity.DEFAULT);
479480
}
480481

481482
/**
@@ -492,12 +493,13 @@ public TimeSeriesOptions metaField(String metaField) {
492493
}
493494

494495
/**
495-
* Select the {@link Granularity} parameter to define how data in the time series collection is organized. Select
496-
* one that is closest to the time span between incoming measurements.
496+
* Select the {@link GranularityDefinition} parameter to define how data in the time series collection is organized.
497+
* Select one that is closest to the time span between incoming measurements.
497498
*
498499
* @return new instance of {@link TimeSeriesOptions}.
500+
* @see Granularity
499501
*/
500-
public TimeSeriesOptions granularity(Granularity granularity) {
502+
public TimeSeriesOptions granularity(GranularityDefinition granularity) {
501503
return new TimeSeriesOptions(timeField, metaField, granularity);
502504
}
503505

@@ -520,7 +522,7 @@ public String getMetaField() {
520522
/**
521523
* @return never {@literal null}.
522524
*/
523-
public Granularity getGranularity() {
525+
public GranularityDefinition getGranularity() {
524526
return granularity;
525527
}
526528
}

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

+14-2
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@
3838
import org.springframework.data.mongodb.core.query.Collation;
3939
import org.springframework.data.mongodb.core.query.Criteria;
4040
import org.springframework.data.mongodb.core.query.Query;
41-
import org.springframework.data.mongodb.core.timeseries.Granularities;
41+
import org.springframework.data.mongodb.core.timeseries.Granularity;
4242
import org.springframework.lang.Nullable;
4343
import org.springframework.util.Assert;
4444
import org.springframework.util.ClassUtils;
@@ -898,11 +898,23 @@ public CollectionOptions getCollectionOptions() {
898898
if (entity.isAnnotationPresent(TimeSeries.class)) {
899899

900900
TimeSeries timeSeries = entity.getRequiredAnnotation(TimeSeries.class);
901+
902+
if (entity.getPersistentProperty(timeSeries.timeField()) == null) {
903+
throw new MappingException(String.format("Time series field '%s' does not exist in type %s",
904+
timeSeries.timeField(), entity.getName()));
905+
}
906+
901907
TimeSeriesOptions options = TimeSeriesOptions.timeSeries(timeSeries.timeField());
902908
if (StringUtils.hasText(timeSeries.metaField())) {
909+
910+
if (entity.getPersistentProperty(timeSeries.metaField()) == null) {
911+
throw new MappingException(
912+
String.format("Meta field '%s' does not exist in type %s", timeSeries.metaField(), entity.getName()));
913+
}
914+
903915
options = options.metaField(timeSeries.metaField());
904916
}
905-
if (!Granularities.DEFAULT.equals(timeSeries.granularity())) {
917+
if (!Granularity.DEFAULT.equals(timeSeries.granularity())) {
906918
options = options.granularity(timeSeries.granularity());
907919
}
908920
collectionOptions = collectionOptions.timeSeries(options);

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

+30-28
Original file line numberDiff line numberDiff line change
@@ -99,7 +99,7 @@
9999
import org.springframework.data.mongodb.core.query.Query;
100100
import org.springframework.data.mongodb.core.query.UpdateDefinition;
101101
import org.springframework.data.mongodb.core.query.UpdateDefinition.ArrayFilter;
102-
import org.springframework.data.mongodb.core.timeseries.Granularities;
102+
import org.springframework.data.mongodb.core.timeseries.Granularity;
103103
import org.springframework.data.mongodb.core.validation.Validator;
104104
import org.springframework.data.mongodb.util.BsonUtils;
105105
import org.springframework.data.projection.SpelAwareProxyProjectionFactory;
@@ -2436,14 +2436,15 @@ protected MongoCollection<Document> doCreateCollection(String collectionName, Do
24362436
co.validationOptions(options);
24372437
}
24382438

2439-
if(collectionOptions.containsKey("timeseries")) {
2439+
if (collectionOptions.containsKey("timeseries")) {
24402440

24412441
Document timeSeries = collectionOptions.get("timeseries", Document.class);
2442-
com.mongodb.client.model.TimeSeriesOptions options = new com.mongodb.client.model.TimeSeriesOptions(timeSeries.getString("timeField"));
2443-
if(timeSeries.containsKey("metaField")) {
2442+
com.mongodb.client.model.TimeSeriesOptions options = new com.mongodb.client.model.TimeSeriesOptions(
2443+
timeSeries.getString("timeField"));
2444+
if (timeSeries.containsKey("metaField")) {
24442445
options.metaField(timeSeries.getString("metaField"));
24452446
}
2446-
if(timeSeries.containsKey("granularity")) {
2447+
if (timeSeries.containsKey("granularity")) {
24472448
options.granularity(TimeSeriesGranularity.valueOf(timeSeries.getString("granularity").toUpperCase()));
24482449
}
24492450
co.timeSeriesOptions(options);
@@ -2604,17 +2605,18 @@ protected Document convertToDocument(@Nullable CollectionOptions collectionOptio
26042605
collectionOptions.getValidationOptions().ifPresent(it -> it.getValidator() //
26052606
.ifPresent(val -> doc.put("validator", getMappedValidator(val, targetType))));
26062607

2607-
collectionOptions.getTimeSeriesOptions().map(operations.forType(targetType)::mapTimeSeriesOptions).ifPresent(it -> {
2608+
collectionOptions.getTimeSeriesOptions().map(operations.forType(targetType)::mapTimeSeriesOptions)
2609+
.ifPresent(it -> {
26082610

2609-
Document timeseries = new Document("timeField", it.getTimeField());
2610-
if(StringUtils.hasText(it.getMetaField())) {
2611-
timeseries.append("metaField", it.getMetaField());
2612-
}
2613-
if(!Granularities.DEFAULT.equals(it.getGranularity())) {
2614-
timeseries.append("granularity", it.getGranularity().name().toLowerCase());
2615-
}
2616-
doc.put("timeseries", timeseries);
2617-
});
2611+
Document timeseries = new Document("timeField", it.getTimeField());
2612+
if (StringUtils.hasText(it.getMetaField())) {
2613+
timeseries.append("metaField", it.getMetaField());
2614+
}
2615+
if (!Granularity.DEFAULT.equals(it.getGranularity())) {
2616+
timeseries.append("granularity", it.getGranularity().name().toLowerCase());
2617+
}
2618+
doc.put("timeseries", timeseries);
2619+
});
26182620
}
26192621

26202622
return doc;
@@ -2849,9 +2851,9 @@ private void executeQueryInternal(CollectionCallback<FindIterable<Document>> col
28492851
.initiateFind(getAndPrepareCollection(doGetDatabase(), collectionName), collectionCallback::doInCollection)
28502852
.iterator()) {
28512853

2852-
while (cursor.hasNext()) {
2853-
callbackHandler.processDocument(cursor.next());
2854-
}
2854+
while (cursor.hasNext()) {
2855+
callbackHandler.processDocument(cursor.next());
2856+
}
28552857
} catch (RuntimeException e) {
28562858
throw potentiallyConvertRuntimeException(e, exceptionTranslator);
28572859
}
@@ -3175,17 +3177,17 @@ private class ReadDocumentCallback<T> implements DocumentCallback<T> {
31753177

31763178
public T doWith(Document document) {
31773179

3178-
maybeEmitEvent(new AfterLoadEvent<>(document, type, collectionName));
3179-
T entity = reader.read(type, document);
3180+
maybeEmitEvent(new AfterLoadEvent<>(document, type, collectionName));
3181+
T entity = reader.read(type, document);
31803182

3181-
if (entity == null) {
3182-
throw new MappingException(String.format("EntityReader %s returned null", reader));
3183-
}
3183+
if (entity == null) {
3184+
throw new MappingException(String.format("EntityReader %s returned null", reader));
3185+
}
31843186

3185-
maybeEmitEvent(new AfterConvertEvent<>(document, entity, collectionName));
3186-
entity = maybeCallAfterConvert(entity, document, collectionName);
3187+
maybeEmitEvent(new AfterConvertEvent<>(document, entity, collectionName));
3188+
entity = maybeCallAfterConvert(entity, document, collectionName);
31873189

3188-
return entity;
3190+
return entity;
31893191
}
31903192
}
31913193

@@ -3237,8 +3239,8 @@ public T doWith(Document document) {
32373239

32383240
Object result = targetType.isInterface() ? projectionFactory.createProjection(targetType, entity) : entity;
32393241

3240-
maybeEmitEvent(new AfterConvertEvent<>(document, result, collectionName));
3241-
return (T) maybeCallAfterConvert(result, document, collectionName);
3242+
maybeEmitEvent(new AfterConvertEvent<>(document, result, collectionName));
3243+
return (T) maybeCallAfterConvert(result, document, collectionName);
32423244
}
32433245
}
32443246

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

+11-11
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,6 @@
1717

1818
import static org.springframework.data.mongodb.core.query.SerializationUtils.*;
1919

20-
import org.springframework.data.mongodb.core.timeseries.Granularities;
2120
import reactor.core.publisher.Flux;
2221
import reactor.core.publisher.Mono;
2322
import reactor.util.function.Tuple2;
@@ -111,6 +110,7 @@
111110
import org.springframework.data.mongodb.core.query.Query;
112111
import org.springframework.data.mongodb.core.query.UpdateDefinition;
113112
import org.springframework.data.mongodb.core.query.UpdateDefinition.ArrayFilter;
113+
import org.springframework.data.mongodb.core.timeseries.Granularity;
114114
import org.springframework.data.mongodb.core.validation.Validator;
115115
import org.springframework.data.mongodb.util.BsonUtils;
116116
import org.springframework.data.projection.SpelAwareProxyProjectionFactory;
@@ -975,7 +975,8 @@ public <O> Flux<O> aggregate(Aggregation aggregation, String collectionName, Cla
975975
return doAggregate(aggregation, collectionName, null, outputType);
976976
}
977977

978-
protected <O> Flux<O> doAggregate(Aggregation aggregation, String collectionName, @Nullable Class<?> inputType, Class<O> outputType) {
978+
protected <O> Flux<O> doAggregate(Aggregation aggregation, String collectionName, @Nullable Class<?> inputType,
979+
Class<O> outputType) {
979980

980981
Assert.notNull(aggregation, "Aggregation pipeline must not be null!");
981982
Assert.hasText(collectionName, "Collection name must not be null or empty!");
@@ -987,19 +988,18 @@ protected <O> Flux<O> doAggregate(Aggregation aggregation, String collectionName
987988
AggregationDefinition ctx = queryOperations.createAggregation(aggregation, inputType);
988989

989990
if (LOGGER.isDebugEnabled()) {
990-
LOGGER.debug("Streaming aggregation: {} in collection {}", serializeToJsonSafely(ctx.getAggregationPipeline()), collectionName);
991+
LOGGER.debug("Streaming aggregation: {} in collection {}", serializeToJsonSafely(ctx.getAggregationPipeline()),
992+
collectionName);
991993
}
992994

993995
ReadDocumentCallback<O> readCallback = new ReadDocumentCallback<>(mongoConverter, outputType, collectionName);
994-
return execute(collectionName,
995-
collection -> aggregateAndMap(collection, ctx.getAggregationPipeline(), ctx.isOutOrMerge(), options,
996-
readCallback,
997-
ctx.getInputType()));
996+
return execute(collectionName, collection -> aggregateAndMap(collection, ctx.getAggregationPipeline(),
997+
ctx.isOutOrMerge(), options, readCallback, ctx.getInputType()));
998998
}
999999

10001000
private <O> Flux<O> aggregateAndMap(MongoCollection<Document> collection, List<Document> pipeline,
1001-
boolean isOutOrMerge,
1002-
AggregationOptions options, ReadDocumentCallback<O> readCallback, @Nullable Class<?> inputType) {
1001+
boolean isOutOrMerge, AggregationOptions options, ReadDocumentCallback<O> readCallback,
1002+
@Nullable Class<?> inputType) {
10031003

10041004
AggregatePublisher<Document> cursor = collection.aggregate(pipeline, Document.class)
10051005
.allowDiskUse(options.isAllowDiskUse());
@@ -2510,10 +2510,10 @@ protected CreateCollectionOptions convertToCreateCollectionOptions(@Nullable Col
25102510

25112511
TimeSeriesOptions options = new TimeSeriesOptions(it.getTimeField());
25122512

2513-
if(StringUtils.hasText(it.getMetaField())) {
2513+
if (StringUtils.hasText(it.getMetaField())) {
25142514
options.metaField(it.getMetaField());
25152515
}
2516-
if(!Granularities.DEFAULT.equals(it.getGranularity())) {
2516+
if (!Granularity.DEFAULT.equals(it.getGranularity())) {
25172517
options.granularity(TimeSeriesGranularity.valueOf(it.getGranularity().name().toUpperCase()));
25182518
}
25192519

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

+9-8
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@
2222
import java.lang.annotation.Target;
2323

2424
import org.springframework.core.annotation.AliasFor;
25-
import org.springframework.data.mongodb.core.timeseries.Granularities;
25+
import org.springframework.data.mongodb.core.timeseries.Granularity;
2626

2727
/**
2828
* Identifies a domain object to be persisted to a MongoDB Time Series collection.
@@ -50,8 +50,9 @@
5050
String collection() default "";
5151

5252
/**
53-
* The name of the property which contains the date in each time series document. <br />
54-
* {@link Field#name() Annotated fieldnames} will be considered during the mapping process.
53+
* Name of the property which contains the date in each time series document. <br />
54+
* Translation of property names to {@link Field#name() annotated fieldnames} will be considered during the mapping
55+
* process.
5556
*
5657
* @return never {@literal null}.
5758
*/
@@ -60,19 +61,19 @@
6061
/**
6162
* The name of the field which contains metadata in each time series document. Should not be the {@literal id} nor
6263
* {@link #timeField()} nor point to an {@literal array} or {@link java.util.Collection}. <br />
63-
* {@link Field#name() Annotated fieldnames} will be considered during the mapping process.
64+
* Translation of property names to {@link Field#name() annotated fieldnames} will be considered during the mapping
65+
* process.
6466
*
6567
* @return empty {@link String} by default.
6668
*/
6769
String metaField() default "";
6870

6971
/**
70-
* Select the {@link Granularities granularity} parameter to define how data in the time series collection is
71-
* organized.
72+
* Select the {@link Granularity granularity} parameter to define how data in the time series collection is organized.
7273
*
73-
* @return {@link Granularities#DEFAULT server default} by default.
74+
* @return {@link Granularity#DEFAULT server default} by default.
7475
*/
75-
Granularities granularity() default Granularities.DEFAULT;
76+
Granularity granularity() default Granularity.DEFAULT;
7677

7778
/**
7879
* Defines the collation to apply when executing a query or creating indexes.

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

+21-3
Original file line numberDiff line numberDiff line change
@@ -16,12 +16,30 @@
1616
package org.springframework.data.mongodb.core.timeseries;
1717

1818
/**
19-
* The Granularity of time series data that is closest to the time span between incoming measurements.
19+
* {@link GranularityDefinition Granularities} available for Time Series data.
2020
*
2121
* @author Christoph Strobl
2222
* @since 3.3
2323
*/
24-
public interface Granularity {
24+
public enum Granularity implements GranularityDefinition {
2525

26-
String name();
26+
/**
27+
* Server default value to indicate no explicit value should be sent.
28+
*/
29+
DEFAULT,
30+
31+
/**
32+
* High frequency ingestion.
33+
*/
34+
SECONDS,
35+
36+
/**
37+
* Medium frequency ingestion.
38+
*/
39+
MINUTES,
40+
41+
/**
42+
* Low frequency ingestion.
43+
*/
44+
HOURS
2745
}

spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/timeseries/Granularities.java spring-data-mongodb/src/main/java/org/springframework/data/mongodb/core/timeseries/GranularityDefinition.java

+3-21
Original file line numberDiff line numberDiff line change
@@ -16,30 +16,12 @@
1616
package org.springframework.data.mongodb.core.timeseries;
1717

1818
/**
19-
* {@link Granularity Granularities} available for Time Series data.
19+
* The Granularity of time series data that is closest to the time span between incoming measurements.
2020
*
2121
* @author Christoph Strobl
2222
* @since 3.3
2323
*/
24-
public enum Granularities implements Granularity {
24+
public interface GranularityDefinition {
2525

26-
/**
27-
* Server default value to indicate no explicit value should be sent.
28-
*/
29-
DEFAULT,
30-
31-
/**
32-
* High frequency ingestion.
33-
*/
34-
SECONDS,
35-
36-
/**
37-
* Medium frequency ingestion.
38-
*/
39-
MINUTES,
40-
41-
/**
42-
* Low frequency ingestion.
43-
*/
44-
HOURS
26+
String name();
4527
}

0 commit comments

Comments
 (0)