20
20
import java .util .Collections ;
21
21
import java .util .EnumSet ;
22
22
import java .util .List ;
23
+ import java .util .function .Predicate ;
24
+ import java .util .stream .Collectors ;
23
25
26
+ import org .bson .Document ;
24
27
import org .springframework .data .mapping .PersistentProperty ;
25
28
import org .springframework .data .mapping .context .MappingContext ;
26
29
import org .springframework .data .mongodb .core .convert .MongoConverter ;
30
+ import org .springframework .data .mongodb .core .mapping .BasicMongoPersistentEntity ;
31
+ import org .springframework .data .mongodb .core .mapping .Encrypted ;
27
32
import org .springframework .data .mongodb .core .mapping .Field ;
28
33
import org .springframework .data .mongodb .core .mapping .MongoPersistentEntity ;
29
34
import org .springframework .data .mongodb .core .mapping .MongoPersistentProperty ;
35
+ import org .springframework .data .mongodb .core .schema .IdentifiableJsonSchemaProperty .EncryptedJsonSchemaProperty ;
30
36
import org .springframework .data .mongodb .core .schema .IdentifiableJsonSchemaProperty .ObjectJsonSchemaProperty ;
31
37
import org .springframework .data .mongodb .core .schema .JsonSchemaObject ;
32
38
import org .springframework .data .mongodb .core .schema .JsonSchemaObject .Type ;
33
39
import org .springframework .data .mongodb .core .schema .JsonSchemaProperty ;
34
40
import org .springframework .data .mongodb .core .schema .MongoJsonSchema ;
35
41
import org .springframework .data .mongodb .core .schema .MongoJsonSchema .MongoJsonSchemaBuilder ;
36
42
import org .springframework .data .mongodb .core .schema .TypedJsonSchemaObject ;
43
+ import org .springframework .lang .Nullable ;
37
44
import org .springframework .util .Assert ;
38
45
import org .springframework .util .ClassUtils ;
39
46
import org .springframework .util .CollectionUtils ;
40
47
import org .springframework .util .ObjectUtils ;
48
+ import org .springframework .util .StringUtils ;
41
49
42
50
/**
43
51
* {@link MongoJsonSchemaCreator} implementation using both {@link MongoConverter} and {@link MappingContext} to obtain
@@ -52,6 +60,7 @@ class MappingMongoJsonSchemaCreator implements MongoJsonSchemaCreator {
52
60
53
61
private final MongoConverter converter ;
54
62
private final MappingContext <MongoPersistentEntity <?>, MongoPersistentProperty > mappingContext ;
63
+ private final Predicate <JsonSchemaPropertyContext > filter ;
55
64
56
65
/**
57
66
* Create a new instance of {@link MappingMongoJsonSchemaCreator}.
@@ -61,10 +70,24 @@ class MappingMongoJsonSchemaCreator implements MongoJsonSchemaCreator {
61
70
@ SuppressWarnings ("unchecked" )
62
71
MappingMongoJsonSchemaCreator (MongoConverter converter ) {
63
72
73
+ this (converter , (MappingContext <MongoPersistentEntity <?>, MongoPersistentProperty >) converter .getMappingContext (),
74
+ (property ) -> true );
75
+ }
76
+
77
+ @ SuppressWarnings ("unchecked" )
78
+ MappingMongoJsonSchemaCreator (MongoConverter converter ,
79
+ MappingContext <MongoPersistentEntity <?>, MongoPersistentProperty > mappingContext ,
80
+ Predicate <JsonSchemaPropertyContext > filter ) {
81
+
64
82
Assert .notNull (converter , "Converter must not be null!" );
65
83
this .converter = converter ;
66
- this .mappingContext = (MappingContext <MongoPersistentEntity <?>, MongoPersistentProperty >) converter
67
- .getMappingContext ();
84
+ this .mappingContext = mappingContext ;
85
+ this .filter = filter ;
86
+ }
87
+
88
+ @ Override
89
+ public MongoJsonSchemaCreator filter (Predicate <JsonSchemaPropertyContext > filter ) {
90
+ return new MappingMongoJsonSchemaCreator (converter , mappingContext , filter );
68
91
}
69
92
70
93
/*
@@ -77,11 +100,29 @@ public MongoJsonSchema createSchemaFor(Class<?> type) {
77
100
MongoPersistentEntity <?> entity = mappingContext .getRequiredPersistentEntity (type );
78
101
MongoJsonSchemaBuilder schemaBuilder = MongoJsonSchema .builder ();
79
102
103
+ {
104
+ Encrypted encrypted = entity .findAnnotation (Encrypted .class );
105
+ if (encrypted != null ) {
106
+
107
+ Document encryptionMetadata = new Document ();
108
+
109
+ Collection <Object > encryptionKeyIds = entity .getEncryptionKeyIds ();
110
+ if (!CollectionUtils .isEmpty (encryptionKeyIds )) {
111
+ encryptionMetadata .append ("keyId" , encryptionKeyIds );
112
+ }
113
+
114
+ if (StringUtils .hasText (encrypted .algorithm ())) {
115
+ encryptionMetadata .append ("algorithm" , encrypted .algorithm ());
116
+ }
117
+
118
+ schemaBuilder .encryptionMetadata (encryptionMetadata );
119
+ }
120
+ }
121
+
80
122
List <JsonSchemaProperty > schemaProperties = computePropertiesForEntity (Collections .emptyList (), entity );
81
123
schemaBuilder .properties (schemaProperties .toArray (new JsonSchemaProperty [0 ]));
82
124
83
125
return schemaBuilder .build ();
84
-
85
126
}
86
127
87
128
private List <JsonSchemaProperty > computePropertiesForEntity (List <MongoPersistentProperty > path ,
@@ -93,6 +134,11 @@ private List<JsonSchemaProperty> computePropertiesForEntity(List<MongoPersistent
93
134
94
135
List <MongoPersistentProperty > currentPath = new ArrayList <>(path );
95
136
137
+ if (!filter .test (new PropertyContext (
138
+ currentPath .stream ().map (PersistentProperty ::getName ).collect (Collectors .joining ("." )), nested ))) {
139
+ continue ;
140
+ }
141
+
96
142
if (path .contains (nested )) { // cycle guard
97
143
schemaProperties .add (createSchemaProperty (computePropertyFieldName (CollectionUtils .lastElement (currentPath )),
98
144
Object .class , false ));
@@ -120,15 +166,38 @@ private JsonSchemaProperty computeSchemaForProperty(List<MongoPersistentProperty
120
166
121
167
String fieldName = computePropertyFieldName (property );
122
168
169
+ JsonSchemaProperty schemaProperty ;
123
170
if (property .isCollectionLike ()) {
124
- return createSchemaProperty (fieldName , targetType , required );
171
+ schemaProperty = createSchemaProperty (fieldName , targetType , required );
125
172
} else if (property .isMap ()) {
126
- return createSchemaProperty (fieldName , Type .objectType (), required );
173
+ schemaProperty = createSchemaProperty (fieldName , Type .objectType (), required );
127
174
} else if (ClassUtils .isAssignable (Enum .class , targetType )) {
128
- return createEnumSchemaProperty (fieldName , targetType , required );
175
+ schemaProperty = createEnumSchemaProperty (fieldName , targetType , required );
176
+ } else {
177
+ schemaProperty = createSchemaProperty (fieldName , targetType , required );
129
178
}
130
179
131
- return createSchemaProperty (fieldName , targetType , required );
180
+ return applyEncryptionDataIfNecessary (property , schemaProperty );
181
+ }
182
+
183
+ @ Nullable
184
+ private JsonSchemaProperty applyEncryptionDataIfNecessary (MongoPersistentProperty property ,
185
+ JsonSchemaProperty schemaProperty ) {
186
+
187
+ Encrypted encrypted = property .findAnnotation (Encrypted .class );
188
+ if (encrypted == null ) {
189
+ return schemaProperty ;
190
+ }
191
+
192
+ EncryptedJsonSchemaProperty enc = new EncryptedJsonSchemaProperty (schemaProperty );
193
+ if (StringUtils .hasText (encrypted .algorithm ())) {
194
+ enc = enc .algorithm (encrypted .algorithm ());
195
+ }
196
+ if (!ObjectUtils .isEmpty (encrypted .keyId ())) {
197
+ enc = enc .keys (property .getEncryptionKeyIds ());
198
+ }
199
+ return enc ;
200
+
132
201
}
133
202
134
203
private JsonSchemaProperty createObjectSchemaPropertyForEntity (List <MongoPersistentProperty > path ,
@@ -207,4 +276,30 @@ static JsonSchemaProperty createPotentiallyRequiredSchemaProperty(JsonSchemaProp
207
276
208
277
return JsonSchemaProperty .required (property );
209
278
}
279
+
280
+ class PropertyContext implements JsonSchemaPropertyContext {
281
+
282
+ private String path ;
283
+ private MongoPersistentProperty property ;
284
+
285
+ public PropertyContext (String path , MongoPersistentProperty property ) {
286
+ this .path = path ;
287
+ this .property = property ;
288
+ }
289
+
290
+ @ Override
291
+ public String getPath () {
292
+ return path ;
293
+ }
294
+
295
+ @ Override
296
+ public MongoPersistentProperty getProperty () {
297
+ return property ;
298
+ }
299
+
300
+ @ Override
301
+ public <T > MongoPersistentEntity <T > resolveEntity (MongoPersistentProperty property ) {
302
+ return (MongoPersistentEntity <T >) mappingContext .getPersistentEntity (property );
303
+ }
304
+ }
210
305
}
0 commit comments