All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.azure.core.models.JsonPatchDocument Maven / Gradle / Ivy

The newest version!
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

package com.azure.core.models;

import com.azure.core.implementation.Option;
import com.azure.core.util.CoreUtils;
import com.azure.core.util.serializer.JacksonAdapter;
import com.azure.core.util.serializer.JsonSerializer;
import com.azure.core.util.serializer.JsonSerializerProviders;
import com.azure.json.JsonReader;
import com.azure.json.JsonSerializable;
import com.azure.json.JsonWriter;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonValue;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;

/**
 * 

Represents a JSON Patch document.

* *

This class encapsulates a list of {@link JsonPatchOperation} instances that form a JSON Patch document. * It provides methods to add various types of operations (add, replace, copy, move, remove, test) to the document.

* *

Each operation in the document is represented by a {@link JsonPatchOperation} instance, which encapsulates the * operation kind, path, and optional from and value.

* *

This class also provides a {@link #toJson(JsonWriter)} method to serialize the JSON Patch document to JSON, * and a {@link #fromJson(JsonReader)} method to deserialize a JSON Patch document from JSON.

* *

This class is useful when you want to create a JSON Patch document to express a sequence of operations to * apply to a JSON document.

* * @see JsonPatchOperation * @see JsonPatchOperationKind * @see JsonSerializable */ public final class JsonPatchDocument implements JsonSerializable { private static final Object SERIALIZER_INSTANTIATION_SYNCHRONIZER = new Object(); private static volatile JsonSerializer defaultSerializer; @JsonIgnore private final JsonSerializer serializer; @JsonValue private final List operations; /** * Creates a new JSON Patch document. */ public JsonPatchDocument() { this(null); } /** * Creates a new JSON Patch document. *

* If {@code serializer} isn't specified {@link JacksonAdapter} will be used. * * @param serializer The {@link JsonSerializer} that will be used to serialize patch operation values. */ public JsonPatchDocument(JsonSerializer serializer) { this.operations = new ArrayList<>(); this.serializer = serializer; } /** * Gets a representation of the {@link JsonPatchOperation JSON patch operations} in this JSON patch document. *

* Modifications to the returned list won't mutate the operations in the document. * * @return The JSON patch operations in this JSON patch document. */ List getOperations() { return new ArrayList<>(operations); } /** * Appends an "add" operation to this JSON Patch document. *

* If the {@code path} doesn't exist a new member is added to the object. If the {@code path} does exist the * previous value is replaced. If the {@code path} specifies an array index the value is inserted at the specified. *

* See JSON Patch Add for more information. * *

Code Samples

* * *
     * /*
     *  * Add an object member to the JSON document { "foo" : "bar" } to get the JSON document
     *  * { "bar": "foo", "foo": "bar" }.
     *  */
     * jsonPatchDocument.appendAdd("/bar", "foo");
     *
     * /*
     *  * Add an array element to the JSON document { "foo": [ "fizz", "fizzbuzz" ] } to get the JSON document
     *  * { "foo": [ "fizz", "buzz", "fizzbuzz" ] }.
     *  */
     * jsonPatchDocument.appendAdd("/foo/1", "buzz");
     *
     * /*
     *  * Add a nested member to the JSON document { "foo": "bar" } to get the JSON document
     *  * { "foo": "bar", "child": { "grandchild": { } } }.
     *  */
     * jsonPatchDocument.appendAdd("/child", Collections.singletonMap("grandchild", Collections.emptyMap()));
     *
     * /*
     *  * Add an array element to the JSON document { "foo": [ "fizz", "buzz" ] } to get the JSON document
     *  * { "foo": [ "fizz", "buzz", "fizzbuzz" ] }.
     *  */
     * jsonPatchDocument.appendAdd("/foo/-", "fizzbuzz");
     * 
* * * @param path The path to apply the addition. * @param value The value that will be serialized and added to the path. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendAdd(String path, Object value) { return appendAddInternal(path, serializeValue(value)); } /** * Appends an "add" operation to this JSON Patch document. *

* If the {@code path} doesn't exist a new member is added to the object. If the {@code path} does exist the * previous value is replaced. If the {@code path} specifies an array index the value is inserted at the specified. *

* See JSON Patch Add for more information. * *

Code Samples

* * *
     * /*
     *  * Add an object member to the JSON document { "foo" : "bar" } to get the JSON document
     *  * { "bar": "foo", "foo": "bar" }.
     *  */
     * jsonPatchDocument.appendAddRaw("/bar", "\"foo\"");
     *
     * /*
     *  * Add an array element to the JSON document { "foo": [ "fizz", "fizzbuzz" ] } to get the JSON document
     *  * { "foo": [ "fizz", "buzz", "fizzbuzz" ] }.
     *  */
     * jsonPatchDocument.appendAddRaw("/foo/1", "\"buzz\"");
     *
     * /*
     *  * Add a nested member to the JSON document { "foo": "bar" } to get the JSON document
     *  * { "foo": "bar", "child": { "grandchild": { } } }.
     *  */
     * jsonPatchDocument.appendAddRaw("/child", "\"child\": { \"grandchild\": { } }");
     *
     * /*
     *  * Add an array element to the JSON document { "foo": [ "fizz", "buzz" ] } to get the JSON document
     *  * { "foo": [ "fizz", "buzz", "fizzbuzz" ] }.
     *  */
     * jsonPatchDocument.appendAddRaw("/foo/-", "\"fizzbuzz\"");
     * 
* * * @param path The path to apply the addition. * @param rawJson The raw JSON value that will be added to the path. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendAddRaw(String path, String rawJson) { return appendAddInternal(path, Option.of(rawJson)); } private JsonPatchDocument appendAddInternal(String path, Option rawJsonOption) { Objects.requireNonNull(path, "'path' cannot be null."); return appendOperation(JsonPatchOperationKind.ADD, null, path, rawJsonOption); } /** * Appends a "replace" operation to this JSON Patch document. *

* See JSON Patch replace for more information. * *

Code Samples

* * *
     * /*
     *  * Replace an object member in the JSON document { "bar": "qux", "foo": "bar" } to get the JSON document
     *  * { "bar": "foo", "foo": "bar" }.
     *  */
     * jsonPatchDocument.appendReplace("/bar", "foo");
     *
     * /*
     *  * Replace an object member in the JSON document { "foo": "fizz" } to get the JSON document
     *  * { "foo": [ "fizz", "buzz", "fizzbuzz" ]  }.
     *  */
     * jsonPatchDocument.appendReplace("/foo", new String[] {"fizz", "buzz", "fizzbuzz"});
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an invalid replace operation as the
     *  * target path doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendReplace("/baz", "foo");
     * 
* * * @param path The path to replace. * @param value The value will be serialized and used as the replacement. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendReplace(String path, Object value) { return appendReplaceInternal(path, serializeValue(value)); } /** * Appends a "replace" operation to this JSON Patch document. *

* See JSON Patch replace for more information. * *

Code Samples

* * *
     * /*
     *  * Replace an object member in the JSON document { "bar": "qux", "foo": "bar" } to get the JSON document
     *  * { "bar": "foo", "foo": "bar" }.
     *  */
     * jsonPatchDocument.appendReplaceRaw("/bar", "\"foo\"");
     *
     * /*
     *  * Replace an object member in the JSON document { "foo": "fizz" } to get the JSON document
     *  * { "foo": [ "fizz", "buzz", "fizzbuzz" ]  }.
     *  */
     * jsonPatchDocument.appendReplaceRaw("/foo", "[ \"fizz\", \"buzz\", \"fizzbuzz\" ]");
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an invalid replace operation as the
     *  * target path doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendReplaceRaw("/baz", "\"foo\"");
     * 
* * * @param path The path to replace. * @param rawJson The raw JSON value that will be used as the replacement. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendReplaceRaw(String path, String rawJson) { return appendReplaceInternal(path, Option.of(rawJson)); } private JsonPatchDocument appendReplaceInternal(String path, Option rawJsonOption) { Objects.requireNonNull(path, "'path' cannot be null."); return appendOperation(JsonPatchOperationKind.REPLACE, null, path, rawJsonOption); } /** * Appends a "copy" operation to this JSON Patch document. *

* See JSON Patch copy for more information. * *

Code Samples

* * *
     * /*
     *  * Copy an object member in the JSON document { "foo": "bar" } to get the JSON document
     *  * { "foo": "bar", "copy": "bar" }.
     *  */
     * jsonPatchDocument.appendCopy("/foo", "/copy");
     *
     * /*
     *  * Copy an object member in the JSON document { "foo": { "bar": "baz" } } to get the JSON document
     *  * { "foo": { "bar": "baz" }, "bar": "baz" }.
     *  */
     * jsonPatchDocument.appendCopy("/foo/bar", "/bar");
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an invalid copy operation as the
     *  * target from doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendCopy("/baz", "/fizz");
     * 
* * * @param from The path to copy from. * @param path The path to copy to. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code from} or {@code path} is null. */ public JsonPatchDocument appendCopy(String from, String path) { Objects.requireNonNull(from, "'from' cannot be null."); Objects.requireNonNull(path, "'path' cannot be null."); return appendOperation(JsonPatchOperationKind.COPY, from, path, Option.uninitialized()); } /** * Appends a "move" operation to this JSON Patch document. *

* For the operation to be successful {@code path} cannot be a child node of {@code from}. *

* See JSON Patch move for more information. * *

Code Samples

* * *
     * /*
     *  * Move an object member in the JSON document { "foo": "bar", "bar": "foo" } to get the JSON document
     *  * { "bar": "bar" }.
     *  */
     * jsonPatchDocument.appendMove("/foo", "/bar");
     *
     * /*
     *  * Move an object member in the JSON document { "foo": { "bar": "baz" } } to get the JSON document
     *  * { "foo": "baz" }.
     *  */
     * jsonPatchDocument.appendMove("/foo/bar", "/foo");
     *
     * /*
     *  * Given the JSON document { "foo": { "bar": "baz" } } the following is an example of an invalid move operation
     *  * as the target path is a child of the target from.
     *  */
     * jsonPatchDocument.appendMove("/foo", "/foo/bar");
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an invalid move operation as the
     *  * target from doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendMove("/baz", "/fizz");
     * 
* * * @param from The path to move from. * @param path The path to move to. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code from} or {@code path} is null. */ public JsonPatchDocument appendMove(String from, String path) { Objects.requireNonNull(from, "'from' cannot be null."); Objects.requireNonNull(path, "'path' cannot be null."); return appendOperation(JsonPatchOperationKind.MOVE, from, path, Option.uninitialized()); } /** * Appends a "remove" operation to this JSON Patch document. *

* See JSON Patch remove for more information. * *

Code Samples

* * *
     * /*
     *  * Remove an object member in the JSON document { "foo": "bar", "bar": "foo" } to get the JSON document
     *  * { "foo": "bar" }.
     *  */
     * jsonPatchDocument.appendRemove("/bar");
     *
     * /*
     *  * Remove an object member in the JSON document { "foo": { "bar": "baz" } } to get the JSON document
     *  * { "foo": { } }.
     *  */
     * jsonPatchDocument.appendRemove("/foo/bar");
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an invalid remove operation as the
     *  * target from doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendRemove("/baz");
     * 
* * * @param path The path to remove. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendRemove(String path) { Objects.requireNonNull(path, "'path' cannot be null."); return appendOperation(JsonPatchOperationKind.REMOVE, null, path, Option.uninitialized()); } /** * Appends a "test" operation to this JSON Patch document. *

* See JSON Patch test for more information. * *

Code Samples

* * *
     * /*
     *  * Test an object member in the JSON document { "foo": "bar" } to get a successful operation.
     *  */
     * jsonPatchDocument.appendTest("/foo", "bar");
     *
     * /*
     *  * Test an object member in the JSON document { "foo": "bar" } to get a unsuccessful operation.
     *  */
     * jsonPatchDocument.appendTest("/foo", 42);
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an unsuccessful test operation as
     *  * the target path doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendTest("/baz", "bar");
     * 
* * * @param path The path to test. * @param value The value that will be serialized and used to test against. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendTest(String path, Object value) { return appendTestInternal(path, serializeValue(value)); } /** * Appends a "test" operation to this JSON Patch document. *

* See JSON Patch test for more information. * *

Code Samples

* * *
     * /*
     *  * Test an object member in the JSON document { "foo": "bar" } to get a successful operation.
     *  */
     * jsonPatchDocument.appendTestRaw("/foo", "\"bar\"");
     *
     * /*
     *  * Test an object member in the JSON document { "foo": "bar" } to get a unsuccessful operation.
     *  */
     * jsonPatchDocument.appendTestRaw("/foo", "42");
     *
     * /*
     *  * Given the JSON document { "foo": "bar" } the following is an example of an unsuccessful test operation as
     *  * the target path doesn't exist in the document.
     *  */
     * jsonPatchDocument.appendTestRaw("/baz", "\"bar\"");
     * 
* * * @param path The path to test. * @param rawJson The raw JSON value that will be used to test against. * @return The updated JsonPatchDocument object. * @throws NullPointerException If {@code path} is null. */ public JsonPatchDocument appendTestRaw(String path, String rawJson) { return appendTestInternal(path, Option.of(rawJson)); } private JsonPatchDocument appendTestInternal(String path, Option rawJsonOption) { Objects.requireNonNull(path, "'path' cannot be null."); return appendOperation(JsonPatchOperationKind.TEST, null, path, rawJsonOption); } private Option serializeValue(Object value) { if (value == null) { return Option.empty(); } byte[] bytes; if (serializer == null) { if (defaultSerializer == null) { synchronized (SERIALIZER_INSTANTIATION_SYNCHRONIZER) { if (defaultSerializer == null) { defaultSerializer = JsonSerializerProviders.createInstance(); } } } bytes = defaultSerializer.serializeToBytes(value); } else { bytes = serializer.serializeToBytes(value); } return Option.of(new String(bytes, StandardCharsets.UTF_8)); } private JsonPatchDocument appendOperation(JsonPatchOperationKind operationKind, String from, String path, Option optionalValue) { operations.add(new JsonPatchOperation(operationKind, from, path, optionalValue)); return this; } /** * Gets a formatted JSON string representation of this JSON Patch document. * * @return The formatted JSON String representing this JSON Patch document. */ @Override public String toString() { StringBuilder builder = new StringBuilder("["); for (int i = 0; i < operations.size(); i++) { if (i > 0) { builder.append(","); } operations.get(i).buildString(builder); } return builder.append("]").toString(); } @Override public JsonWriter toJson(JsonWriter jsonWriter) throws IOException { return (CoreUtils.isNullOrEmpty(operations)) ? jsonWriter : jsonWriter.writeArray(operations, JsonWriter::writeJson); } /** * Reads a JSON stream into a {@link JsonPatchDocument}. * * @param jsonReader The {@link JsonReader} being read. * @return The {@link JsonPatchDocument} that the JSON stream represented, or null if it pointed to JSON null. * @throws IllegalStateException If the deserialized JSON object was missing any required properties. * @throws IOException If a {@link JsonPatchDocument} fails to be read from the {@code jsonReader}. */ public static JsonPatchDocument fromJson(JsonReader jsonReader) throws IOException { List operations = jsonReader.readArray(JsonPatchOperation::fromJson); JsonPatchDocument document = new JsonPatchDocument(); if (operations != null) { document.operations.addAll(operations); } return document; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy