com.github.fge.jsonpatch.mergepatch.ObjectMergePatch Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of json-patch Show documentation
Show all versions of json-patch Show documentation
JSON Patch (RFC 6902) and JSON Merge Patch (RFC 7386) implementation in Java
The newest version!
/*
* Copyright (c) 2014, Francis Galiegue ([email protected])
*
* This software is dual-licensed under:
*
* - the Lesser General Public License (LGPL) version 3.0 or, at your option, any
* later version;
* - the Apache Software License (ASL) version 2.0.
*
* The text of this file and of both licenses is available at the root of this
* project or, if you have the jar distribution, in directory META-INF/, under
* the names LGPL-3.0.txt and ASL-2.0.txt respectively.
*
* Direct link to the sources:
*
* - LGPL 3.0: https://www.gnu.org/licenses/lgpl-3.0.txt
* - ASL 2.0: http://www.apache.org/licenses/LICENSE-2.0.txt
*/
package com.github.fge.jsonpatch.mergepatch;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.jsontype.TypeSerializer;
import com.fasterxml.jackson.databind.node.NullNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.github.fge.jackson.JacksonUtils;
import com.github.fge.jsonpatch.JsonPatchException;
import javax.annotation.ParametersAreNonnullByDefault;
import java.io.IOException;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
@ParametersAreNonnullByDefault
final class ObjectMergePatch
extends JsonMergePatch
{
private final Set removedMembers;
private final Map modifiedMembers;
ObjectMergePatch(final Set removedMembers,
final Map modifiedMembers)
{
this.removedMembers = Collections.unmodifiableSet(new HashSet(removedMembers));
this.modifiedMembers = Collections.unmodifiableMap(new HashMap(modifiedMembers));
}
@Override
public JsonNode apply(final JsonNode input)
throws JsonPatchException
{
BUNDLE.checkNotNull(input, "jsonPatch.nullValue");
/*
* If the input is an object, we make a deep copy of it
*/
final ObjectNode ret = input.isObject() ? (ObjectNode) input.deepCopy()
: JacksonUtils.nodeFactory().objectNode();
/*
* Our result is now a JSON Object; first, add (or modify) existing
* members in the result
*/
String key;
JsonNode value;
for (final Map.Entry entry:
modifiedMembers.entrySet()) {
key = entry.getKey();
/*
* FIXME: ugly...
*
* We treat missing keys as null nodes; this "works" because in
* the modifiedMembers map, values are JsonMergePatch instances:
*
* * if it is a NonObjectMergePatch, the value is replaced
* unconditionally;
* * if it is an ObjectMergePatch, we get back here; the value will
* be replaced with a JSON Object anyway before being processed.
*/
final JsonNode jsonNode = ret.get(key);
value = jsonNode != null ? jsonNode : NullNode.getInstance();
ret.replace(key, entry.getValue().apply(value));
}
ret.remove(removedMembers);
return ret;
}
@Override
public void serialize(final JsonGenerator jgen,
final SerializerProvider provider)
throws IOException, JsonProcessingException
{
jgen.writeStartObject();
/*
* Write removed members as JSON nulls
*/
for (final String member: removedMembers)
jgen.writeNullField(member);
/*
* Write modified members; delegate to serialization for writing values
*/
for (final Map.Entry entry:
modifiedMembers.entrySet()) {
jgen.writeFieldName(entry.getKey());
entry.getValue().serialize(jgen, provider);
}
jgen.writeEndObject();
}
@Override
public void serializeWithType(final JsonGenerator jgen,
final SerializerProvider provider, final TypeSerializer typeSer)
throws IOException, JsonProcessingException
{
serialize(jgen, provider);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy