Skip to content

Commit bca25c6

Browse files
committed
#1939 fix merge / PATCH regex based deletion during patch
* 2 problems fixed: * deletion was not applied correctly for nested JsonObjects * chosen regex delimiters `/` were incompatible with HTTP API checks that no slashes may be contained in JsonKeys and feature ids Signed-off-by: Thomas Jäckle <[email protected]>
1 parent 79cafdc commit bca25c6

File tree

3 files changed

+78
-11
lines changed

3 files changed

+78
-11
lines changed

documentation/src/main/resources/pages/ditto/httpapi-concepts.md

+14-3
Original file line numberDiff line numberDiff line change
@@ -408,14 +408,25 @@ JSON objects using a regular expression **before** applying all other patch valu
408408

409409
The syntax for this function is rather specific (so that no "normally" occurring JSON keys match the same syntax):
410410
```json
411+
{
412+
"{%raw%}{{ ~.*~ }}{%endraw%}": null
413+
}
414+
```
415+
416+
So the recommended regex delimiter to use is the `~` character, additionally supported delimiters at this point are:
417+
* `~`
418+
* `/` (discouraged due to the special meaning of `/` for HTTP paths)
419+
420+
A discouraged and thus deprecated way to define the regex (using `/` as delimiter) is:
421+
```json
411422
{
412423
"{%raw%}{{ /.*/ }}{%endraw%}": null
413424
}
414425
```
415426

416-
When such a `{%raw%}{{ /<regex>/ }}{%endraw%}` with the value `null` is detected in the merge patch, the content between the 2 `/` is
427+
When such a `{%raw%}{{ ~<regex>~ }}{%endraw%}` with the value `null` is detected in the merge patch, the content between the 2 delimiters is
417428
interpreted as regular expression to apply for finding keys to delete from the target object.
418-
As a result, using `"{%raw%}{{ /.*/ }}{%endraw%}": null` would delete all the values inside a JSON object before applying the new
429+
As a result, using `"{%raw%}{{ ~.*~ }}{%endraw%}": null` would delete all the values inside a JSON object before applying the new
419430
values provided in the patch.
420431

421432
Example:
@@ -444,7 +455,7 @@ of this year:
444455
"features": {
445456
"aggregated-history": {
446457
"properties": {
447-
"{%raw%}{{ /2022-.*/ }}{%endraw%}": null,
458+
"{%raw%}{{ ~2022-.*~ }}{%endraw%}": null,
448459
"2023-03": 105.21
449460
}
450461
}

json/src/main/java/org/eclipse/ditto/json/JsonMergePatch.java

+14-3
Original file line numberDiff line numberDiff line change
@@ -194,8 +194,10 @@ private static JsonObject mergeJsonObjects(final JsonObject jsonObject1, final J
194194

195195
// add fields of jsonObject2 not present in jsonObject1
196196
jsonObject2.forEach(jsonField -> {
197-
if (!jsonObject1.contains(jsonField.getKey()) && toBeNulledKeysByRegex.contains(jsonField.getKey())) {
197+
if (toBeNulledKeysByRegex.contains(jsonField.getKey())) {
198198
builder.remove(jsonField.getKey());
199+
jsonObject1.getValue(jsonField.getKey())
200+
.ifPresent(v -> builder.set(jsonField.getKey(), v));
199201
}
200202
});
201203

@@ -208,11 +210,11 @@ private static List<JsonKey> determineToBeNulledKeysByRegex(
208210

209211
final List<JsonKey> toBeNulledKeysByRegex = new ArrayList<>();
210212
final List<JsonKey> keyRegexes = jsonObject1.getKeys().stream()
211-
.filter(key -> key.toString().startsWith("{{") && key.toString().endsWith("}}"))
213+
.filter(JsonMergePatch::isEnclosedByCurlyBraces)
212214
.collect(Collectors.toList());
213215
keyRegexes.forEach(keyRegex -> {
214216
final String keyRegexWithoutCurly = keyRegex.toString().substring(2, keyRegex.length() - 2).trim();
215-
if (keyRegexWithoutCurly.startsWith("/") && keyRegexWithoutCurly.endsWith("/")) {
217+
if (isEnclosedByRegexDelimiter(keyRegexWithoutCurly)) {
216218
final String regexStr = keyRegexWithoutCurly.substring(1, keyRegexWithoutCurly.length() - 1);
217219
final Pattern pattern = Pattern.compile(regexStr);
218220
jsonObject1.getValue(keyRegex)
@@ -227,6 +229,15 @@ private static List<JsonKey> determineToBeNulledKeysByRegex(
227229
return toBeNulledKeysByRegex;
228230
}
229231

232+
private static boolean isEnclosedByCurlyBraces(final JsonKey key) {
233+
return key.toString().startsWith("{{") && key.toString().endsWith("}}");
234+
}
235+
236+
private static boolean isEnclosedByRegexDelimiter(final String keyRegexWithoutCurly) {
237+
return keyRegexWithoutCurly.startsWith("~") && keyRegexWithoutCurly.endsWith("~") ||
238+
keyRegexWithoutCurly.startsWith("/") && keyRegexWithoutCurly.endsWith("/");
239+
}
240+
230241
/**
231242
* Applies this merge patch on the given json value.
232243
*

json/src/test/java/org/eclipse/ditto/json/JsonMergePatchTest.java

+50-5
Original file line numberDiff line numberDiff line change
@@ -476,21 +476,21 @@ public void removeFieldsUsingRegexWithNullValue() {
476476

477477
final JsonObject objectToPatch = JsonFactory.newObjectBuilder()
478478
.set("a", JsonFactory.newObjectBuilder()
479-
.set("{{ /2023-04-.*/ }}", JsonValue.nullLiteral())
479+
.set("{{ ~2023-04-.*~ }}", JsonValue.nullLiteral())
480480
.set("2023-05-01", JsonValue.of("new"))
481481
.set("2023-05-02", JsonValue.of("catch"))
482482
.set("2023-05-03", JsonValue.of("phrase"))
483483
.build())
484484
.set("b", JsonFactory.newObjectBuilder()
485-
.set("{{ /2023-04-01/ }}", JsonValue.nullLiteral())
486-
.set("{{ /^2023-04-03$/ }}", JsonValue.nullLiteral())
487-
.set("{{ /[0-9]{4}-04-.+4/ }}", JsonValue.nullLiteral())
485+
.set("{{ ~2023-04-01~ }}", JsonValue.nullLiteral())
486+
.set("{{ ~^2023-04-03$~ }}", JsonValue.nullLiteral())
487+
.set("{{ ~[0-9]{4}-04-.+4~ }}", JsonValue.nullLiteral())
488488
.set("2023-05-01", JsonValue.of("new"))
489489
.set("2023-05-02", JsonValue.of("catch"))
490490
.set("2023-05-03", JsonValue.of("phrase"))
491491
.build())
492492
.set("c", JsonFactory.newObjectBuilder()
493-
.set("{{ /.*/ }}", JsonValue.nullLiteral())
493+
.set("{{ ~.*~ }}", JsonValue.nullLiteral())
494494
.set("2023-05-01", JsonValue.of("new"))
495495
.set("2023-05-02", JsonValue.of("catch"))
496496
.set("2023-05-03", JsonValue.of("phrase"))
@@ -523,5 +523,50 @@ public void removeFieldsUsingRegexWithNullValue() {
523523
Assertions.assertThat(mergedObject).isEqualTo(expectedObject);
524524
}
525525

526+
@Test
527+
public void removeFieldsUsingRegexWithNullValueWithHierarchy() {
528+
final JsonObject originalObject = JsonFactory.newObjectBuilder()
529+
.set("first", JsonFactory.newObjectBuilder()
530+
.set("second", JsonFactory.newObjectBuilder()
531+
.set("third", JsonFactory.newObjectBuilder()
532+
.set("something-on-third", "foobar3")
533+
.set("another-on-third", false)
534+
.build())
535+
.set("something-on-second", "foobar2")
536+
.set("another-on-second", false)
537+
.build())
538+
.set("something-on-first", "foobar1")
539+
.set("another-on-first", 42)
540+
.build()
541+
)
542+
.build();
543+
544+
final JsonObject objectToPatch = JsonFactory.newObjectBuilder()
545+
.set("first", JsonFactory.newObjectBuilder()
546+
.set("{{ ~seco.*~ }}", JsonValue.nullLiteral())
547+
.set("second", JsonFactory.newObjectBuilder()
548+
.set("another-on-second", true)
549+
.build()
550+
)
551+
.build()
552+
)
553+
.build();
554+
555+
final JsonValue expectedObject = JsonFactory.newObjectBuilder()
556+
.set("first", JsonFactory.newObjectBuilder()
557+
.set("second", JsonFactory.newObjectBuilder()
558+
.set("another-on-second", true)
559+
.build())
560+
.set("something-on-first", "foobar1")
561+
.set("another-on-first", 42)
562+
.build()
563+
)
564+
.build();
565+
566+
final JsonValue mergedObject = JsonMergePatch.of(objectToPatch).applyOn(originalObject);
567+
568+
Assertions.assertThat(mergedObject).isEqualTo(expectedObject);
569+
}
570+
526571

527572
}

0 commit comments

Comments
 (0)