/*
 * Copyright (c) 2011-2019 Contributors to the Eclipse Foundation
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License 2.0 which is available at
 * http://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
 * which is available at https://www.apache.org/licenses/LICENSE-2.0.
 *
 * SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
 */
package io.vertx.core.json;

import io.vertx.core.buffer.Buffer;
import io.vertx.core.shareddata.Shareable;
import io.vertx.core.shareddata.impl.ClusterSerializable;

import java.time.Instant;
import java.util.*;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;

import static io.vertx.core.json.impl.JsonUtil.*;
import static java.time.format.DateTimeFormatter.ISO_INSTANT;

A representation of a JSON object in Java. Unlike some other languages Java does not have a native understanding of JSON. To enable JSON to be used easily in Vert.x code we use this class to encapsulate the notion of a JSON object. The implementation adheres to the RFC-7493 to support Temporal data types as well as binary data. Please see the documentation for more information.
Author:Tim Fox
/** * A representation of a <a href="http://json.org/">JSON</a> object in Java. * * Unlike some other languages Java does not have a native understanding of JSON. To enable JSON to be used easily * in Vert.x code we use this class to encapsulate the notion of a JSON object. * * The implementation adheres to the <a href="http://rfc-editor.org/rfc/rfc7493.txt">RFC-7493</a> to support Temporal * data types as well as binary data. * * Please see the documentation for more information. * * @author <a href="http://tfox.org">Tim Fox</a> */
public class JsonObject implements Iterable<Map.Entry<String, Object>>, ClusterSerializable, Shareable { private Map<String, Object> map;
Create an instance from a string of JSON
Params:
  • json – the string of JSON
/** * Create an instance from a string of JSON * * @param json the string of JSON */
public JsonObject(String json) { if (json == null) { throw new NullPointerException(); } fromJson(json); if (map == null) { throw new DecodeException("Invalid JSON object: " + json); } }
Create a new, empty instance
/** * Create a new, empty instance */
public JsonObject() { map = new LinkedHashMap<>(); }
Create an instance from a Map. The Map is not copied.
Params:
  • map – the map to create the instance from.
/** * Create an instance from a Map. The Map is not copied. * * @param map the map to create the instance from. */
public JsonObject(Map<String, Object> map) { if (map == null) { throw new NullPointerException(); } this.map = map; }
Create an instance from a buffer.
Params:
  • buf – the buffer to create the instance from.
/** * Create an instance from a buffer. * * @param buf the buffer to create the instance from. */
public JsonObject(Buffer buf) { if (buf == null) { throw new NullPointerException(); } fromBuffer(buf); if (map == null) { throw new DecodeException("Invalid JSON object: " + buf); } }
Create a JsonObject from the fields of a Java object. Faster than calling `new JsonObject(Json.encode(obj))`.
Params:
  • obj – The object to convert to a JsonObject.
Throws:
/** * Create a JsonObject from the fields of a Java object. * Faster than calling `new JsonObject(Json.encode(obj))`. * <p/ * Returns {@code null} when {@code obj} is {@code null}. * * @param obj The object to convert to a JsonObject. * @throws IllegalArgumentException if conversion fails due to an incompatible type. */
@SuppressWarnings("unchecked") public static JsonObject mapFrom(Object obj) { if (obj == null) { return null; } else { return new JsonObject((Map<String, Object>) Json.CODEC.fromValue(obj, Map.class)); } }
Instantiate a Java object from a JsonObject. Faster than calling `Json.decodeValue(Json.encode(jsonObject), type)`.
Params:
  • type – The type to instantiate from the JsonObject.
Throws:
/** * Instantiate a Java object from a JsonObject. * Faster than calling `Json.decodeValue(Json.encode(jsonObject), type)`. * * @param type The type to instantiate from the JsonObject. * @throws IllegalArgumentException if the type cannot be instantiated. */
public <T> T mapTo(Class<T> type) { return Json.CODEC.fromValue(map, type); }
Get the string value with the specified key, special cases are addressed for extended JSON types Instant, byte[] and Enum which can be converted to String.
Params:
  • key – the key to return the value for
Returns:the value string representation or null if no value for that key
/** * Get the string value with the specified key, special cases are addressed for extended JSON types {@code Instant}, * {@code byte[]} and {@code Enum} which can be converted to String. * * @param key the key to return the value for * @return the value string representation or null if no value for that key */
public String getString(String key) { Objects.requireNonNull(key); Object val = map.get(key); if (val == null) { return null; } if (val instanceof Instant) { return ISO_INSTANT.format((Instant) val); } else if (val instanceof byte[]) { return BASE64_ENCODER.encodeToString((byte[]) val); } else if (val instanceof Buffer) { return BASE64_ENCODER.encodeToString(((Buffer) val).getBytes()); } else if (val instanceof Enum) { return ((Enum) val).name(); } else { return val.toString(); } }
Get the Number value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the Number value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a Number */
public Number getNumber(String key) { Objects.requireNonNull(key); return (Number) map.get(key); }
Get the Integer value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the Integer value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not an Integer */
public Integer getInteger(String key) { Objects.requireNonNull(key); Number number = (Number) map.get(key); if (number == null) { return null; } else if (number instanceof Integer) { return (Integer) number; // Avoids unnecessary unbox/box } else { return number.intValue(); } }
Get the Long value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the Long value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a Long */
public Long getLong(String key) { Objects.requireNonNull(key); Number number = (Number) map.get(key); if (number == null) { return null; } else if (number instanceof Long) { return (Long) number; // Avoids unnecessary unbox/box } else { return number.longValue(); } }
Get the Double value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the Double value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a Double */
public Double getDouble(String key) { Objects.requireNonNull(key); Number number = (Number) map.get(key); if (number == null) { return null; } else if (number instanceof Double) { return (Double) number; // Avoids unnecessary unbox/box } else { return number.doubleValue(); } }
Get the Float value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the Float value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a Float */
public Float getFloat(String key) { Objects.requireNonNull(key); Number number = (Number) map.get(key); if (number == null) { return null; } else if (number instanceof Float) { return (Float) number; // Avoids unnecessary unbox/box } else { return number.floatValue(); } }
Get the Boolean value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the Boolean value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a Boolean */
public Boolean getBoolean(String key) { Objects.requireNonNull(key); return (Boolean) map.get(key); }
Get the JsonObject value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the JsonObject value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a JsonObject */
public JsonObject getJsonObject(String key) { Objects.requireNonNull(key); Object val = map.get(key); if (val instanceof Map) { val = new JsonObject((Map) val); } return (JsonObject) val; }
Get the JsonArray value with the specified key
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the JsonArray value with the specified key * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a JsonArray */
public JsonArray getJsonArray(String key) { Objects.requireNonNull(key); Object val = map.get(key); if (val instanceof List) { val = new JsonArray((List) val); } return (JsonArray) val; }
Get the binary value with the specified key. JSON itself has no notion of a binary, this extension complies to the RFC-7493, so this method assumes there is a String value with the key and it contains a Base64 encoded binary, which it decodes if found and returns.
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the binary value with the specified key. * * JSON itself has no notion of a binary, this extension complies to the RFC-7493, so this method assumes there is a * String value with the key and it contains a Base64 encoded binary, which it decodes if found and returns. * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a String * @throws java.lang.IllegalArgumentException if the String value is not a legal Base64 encoded value */
public byte[] getBinary(String key) { Objects.requireNonNull(key); Object val = map.get(key); // no-op if (val == null) { return null; } // no-op if value is already an byte[] if (val instanceof byte[]) { return (byte[]) val; } // unwrap if value is already a Buffer if (val instanceof Buffer) { return ((Buffer) val).getBytes(); } // assume that the value is in String format as per RFC String encoded = (String) val; // parse to proper type return BASE64_DECODER.decode(encoded); }
Get the Buffer value with the specified key. JSON itself has no notion of a binary, this extension complies to the RFC-7493, so this method assumes there is a String value with the key and it contains a Base64 encoded binary, which it decodes if found and returns.
Params:
  • key – the string to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the {@code Buffer} value with the specified key. * * JSON itself has no notion of a binary, this extension complies to the RFC-7493, so this method assumes there is a * String value with the key and it contains a Base64 encoded binary, which it decodes if found and returns. * * @param key the string to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a string * @throws java.lang.IllegalArgumentException if the value is not a legal Base64 encoded string */
public Buffer getBuffer(String key) { Objects.requireNonNull(key); Object val = map.get(key); // no-op if (val == null) { return null; } // no-op if value is already an Buffer if (val instanceof Buffer) { return (Buffer) val; } // wrap if value is already an byte[] if (val instanceof byte[]) { return Buffer.buffer((byte[]) val); } // assume that the value is in String format as per RFC String encoded = (String) val; // parse to proper type return Buffer.buffer(BASE64_DECODER.decode(encoded)); }
Get the instant value with the specified key. JSON itself has no notion of a temporal types, this extension complies to the RFC-7493, so this method assumes there is a String value with the key and it contains an ISO 8601 encoded date and time format such as "2017-04-03T10:25:41Z", which it decodes if found and returns.
Params:
  • key – the key to return the value for
Throws:
Returns:the value or null if no value for that key
/** * Get the instant value with the specified key. * * JSON itself has no notion of a temporal types, this extension complies to the RFC-7493, so this method assumes * there is a String value with the key and it contains an ISO 8601 encoded date and time format * such as "2017-04-03T10:25:41Z", which it decodes if found and returns. * * @param key the key to return the value for * @return the value or null if no value for that key * @throws java.lang.ClassCastException if the value is not a String * @throws java.time.format.DateTimeParseException if the String value is not a legal ISO 8601 encoded value */
public Instant getInstant(String key) { Objects.requireNonNull(key); Object val = map.get(key); // no-op if (val == null) { return null; } // no-op if value is already an Instant if (val instanceof Instant) { return (Instant) val; } // assume that the value is in String format as per RFC String encoded = (String) val; // parse to proper type return Instant.from(ISO_INSTANT.parse(encoded)); }
Get the value with the specified key, as an Object with types respecting the limitations of JSON.
  • Map will be wrapped to JsonObject
  • List will be wrapped to JsonArray
  • Instant will be converted to String
  • byte[] will be converted to String
  • Enum will be converted to String
Params:
  • key – the key to lookup
Returns:the value
/** * Get the value with the specified key, as an Object with types respecting the limitations of JSON. * <ul> * <li>{@code Map} will be wrapped to {@code JsonObject}</li> * <li>{@code List} will be wrapped to {@code JsonArray}</li> * <li>{@code Instant} will be converted to {@code String}</li> * <li>{@code byte[]} will be converted to {@code String}</li> * <li>{@code Enum} will be converted to {@code String}</li> * </ul> * * @param key the key to lookup * @return the value */
public Object getValue(String key) { Objects.requireNonNull(key); return wrapJsonValue(map.get(key)); }
Like getString(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getString(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public String getString(String key, String def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getString(key); } else { return def; } }
Like getNumber(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getNumber(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Number getNumber(String key, Number def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getNumber(key); } else { return def; } }
Like getInteger(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getInteger(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Integer getInteger(String key, Integer def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getInteger(key); } else { return def; } }
Like getLong(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getLong(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Long getLong(String key, Long def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getLong(key); } else { return def; } }
Like getDouble(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getDouble(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Double getDouble(String key, Double def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getDouble(key); } else { return def; } }
Like getFloat(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getFloat(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Float getFloat(String key, Float def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getFloat(key); } else { return def; } }
Like getBoolean(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getBoolean(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Boolean getBoolean(String key, Boolean def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getBoolean(key); } else { return def; } }
Like getJsonObject(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getJsonObject(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public JsonObject getJsonObject(String key, JsonObject def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getJsonObject(key); } else { return def; } }
Like getJsonArray(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getJsonArray(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public JsonArray getJsonArray(String key, JsonArray def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getJsonArray(key); } else { return def; } }
Like getBinary(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getBinary(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public byte[] getBinary(String key, byte[] def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getBinary(key); } else { return def; } }
Like getBuffer(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getBuffer(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Buffer getBuffer(String key, Buffer def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getBuffer(key); } else { return def; } }
Like getInstant(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getInstant(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Instant getInstant(String key, Instant def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getInstant(key); } else { return def; } }
Like getValue(String) but specifying a default value to return if there is no entry.
Params:
  • key – the key to lookup
  • def – the default value to use if the entry is not present
Returns:the value or def if no entry present
/** * Like {@link #getValue(String)} but specifying a default value to return if there is no entry. * * @param key the key to lookup * @param def the default value to use if the entry is not present * @return the value or {@code def} if no entry present */
public Object getValue(String key, Object def) { Objects.requireNonNull(key); if (map.containsKey(key)) { return getValue(key); } else { return def; } }
Does the JSON object contain the specified key?
Params:
  • key – the key
Returns:true if it contains the key, false if not.
/** * Does the JSON object contain the specified key? * * @param key the key * @return true if it contains the key, false if not. */
public boolean containsKey(String key) { Objects.requireNonNull(key); return map.containsKey(key); }
Return the set of field names in the JSON objects
Returns:the set of field names
/** * Return the set of field names in the JSON objects * * @return the set of field names */
public Set<String> fieldNames() { return map.keySet(); }
Put a null value into the JSON object with the specified key.
Params:
  • key – the key
Returns: a reference to this, so the API can be used fluently
/** * Put a null value into the JSON object with the specified key. * * @param key the key * @return a reference to this, so the API can be used fluently */
public JsonObject putNull(String key) { Objects.requireNonNull(key); map.put(key, null); return this; }
Put an Object into the JSON object with the specified key.
Params:
  • key – the key
  • value – the value
Returns:a reference to this, so the API can be used fluently
/** * Put an Object into the JSON object with the specified key. * * @param key the key * @param value the value * @return a reference to this, so the API can be used fluently */
public JsonObject put(String key, Object value) { Objects.requireNonNull(key); map.put(key, value); return this; }
Remove an entry from this object.
Params:
  • key – the key
Returns:the value that was removed, or null if none
/** * Remove an entry from this object. * * @param key the key * @return the value that was removed, or null if none */
public Object remove(String key) { Objects.requireNonNull(key); return wrapJsonValue(map.remove(key)); }
Merge in another JSON object. This is the equivalent of putting all the entries of the other JSON object into this object. This is not a deep merge, entries containing (sub) JSON objects will be replaced entirely.
Params:
  • other – the other JSON object
Returns:a reference to this, so the API can be used fluently
/** * Merge in another JSON object. * * This is the equivalent of putting all the entries of the other JSON object into this object. This is not a deep * merge, entries containing (sub) JSON objects will be replaced entirely. * * @param other the other JSON object * @return a reference to this, so the API can be used fluently */
public JsonObject mergeIn(JsonObject other) { return mergeIn(other, false); }
Merge in another JSON object. A deep merge (recursive) matches (sub) JSON objects in the existing tree and replaces all matching entries. JsonArrays are treated like any other entry, i.e. replaced entirely.
Params:
  • other – the other JSON object
  • deep – if true, a deep merge is performed
Returns:a reference to this, so the API can be used fluently
/** * Merge in another JSON object. * A deep merge (recursive) matches (sub) JSON objects in the existing tree and replaces all * matching entries. JsonArrays are treated like any other entry, i.e. replaced entirely. * * @param other the other JSON object * @param deep if true, a deep merge is performed * @return a reference to this, so the API can be used fluently */
public JsonObject mergeIn(JsonObject other, boolean deep) { return mergeIn(other, deep ? Integer.MAX_VALUE : 1); }
Merge in another JSON object. The merge is deep (recursive) to the specified level. If depth is 0, no merge is performed, if depth is greater than the depth of one of the objects, a full deep merge is performed.
Params:
  • other – the other JSON object
  • depth – depth of merge
Returns:a reference to this, so the API can be used fluently
/** * Merge in another JSON object. * The merge is deep (recursive) to the specified level. If depth is 0, no merge is performed, * if depth is greater than the depth of one of the objects, a full deep merge is performed. * * @param other the other JSON object * @param depth depth of merge * @return a reference to this, so the API can be used fluently */
@SuppressWarnings("unchecked") public JsonObject mergeIn(JsonObject other, int depth) { if (depth < 1) { return this; } if (depth == 1) { map.putAll(other.map); return this; } for (Map.Entry<String, Object> e : other.map.entrySet()) { if (e.getValue() == null) { map.put(e.getKey(), null); } else { map.merge(e.getKey(), e.getValue(), (oldVal, newVal) -> { if (oldVal instanceof Map) { oldVal = new JsonObject((Map) oldVal); } if (newVal instanceof Map) { newVal = new JsonObject((Map) newVal); } if (oldVal instanceof JsonObject && newVal instanceof JsonObject) { return ((JsonObject) oldVal).mergeIn((JsonObject) newVal, depth - 1); } return newVal; }); } } return this; }
Encode this JSON object as a string.
Returns:the string encoding.
/** * Encode this JSON object as a string. * * @return the string encoding. */
public String encode() { return Json.CODEC.toString(this, false); }
Encode this JSON object a a string, with whitespace to make the object easier to read by a human, or other sentient organism.
Returns:the pretty string encoding.
/** * Encode this JSON object a a string, with whitespace to make the object easier to read by a human, or other * sentient organism. * * @return the pretty string encoding. */
public String encodePrettily() { return Json.CODEC.toString(this, true); }
Encode this JSON object as buffer.
Returns:the buffer encoding.
/** * Encode this JSON object as buffer. * * @return the buffer encoding. */
public Buffer toBuffer() { return Json.CODEC.toBuffer(this, false); }
Deep copy of this JSON object.
Returns:a copy where all elements have been copied recursively
/** * Deep copy of this JSON object. * * @return a copy where all elements have been copied recursively */
@Override public JsonObject copy() { Map<String, Object> copiedMap; if (map instanceof LinkedHashMap) { copiedMap = new LinkedHashMap<>(map.size()); } else { copiedMap = new HashMap<>(map.size()); } for (Map.Entry<String, Object> entry : map.entrySet()) { Object val = checkAndCopy(entry.getValue()); copiedMap.put(entry.getKey(), val); } return new JsonObject(copiedMap); }
Get the underlying Map as is. This map may contain values that are not the types returned by the JsonObject and with an unpredictable representation of the value, e.g you might get a JSON object as a JsonObject or as a Map.
Returns:the underlying Map.
/** * Get the underlying {@code Map} as is. * * This map may contain values that are not the types returned by the {@code JsonObject} and * with an unpredictable representation of the value, e.g you might get a JSON object * as a {@link JsonObject} or as a {@link Map}. * * @return the underlying Map. */
public Map<String, Object> getMap() { return map; }
Get a stream of the entries in the JSON object.
Returns:a stream of the entries.
/** * Get a stream of the entries in the JSON object. * * @return a stream of the entries. */
public Stream<Map.Entry<String, Object>> stream() { return asStream(iterator()); }
Get an Iterator of the entries in the JSON object.
Returns:an Iterator of the entries
/** * Get an Iterator of the entries in the JSON object. * * @return an Iterator of the entries */
@Override public Iterator<Map.Entry<String, Object>> iterator() { return new Iter(map.entrySet().iterator()); }
Get the number of entries in the JSON object
Returns:the number of entries
/** * Get the number of entries in the JSON object * * @return the number of entries */
public int size() { return map.size(); }
Remove all the entries in this JSON object
/** * Remove all the entries in this JSON object */
public JsonObject clear() { map.clear(); return this; }
Is this object entry?
Returns:true if it has zero entries, false if not.
/** * Is this object entry? * * @return true if it has zero entries, false if not. */
public boolean isEmpty() { return map.isEmpty(); } @Override public String toString() { return encode(); } @Override public boolean equals(Object o) { // null check if (o == null) return false; // self check if (this == o) return true; // type check and cast if (getClass() != o.getClass()) return false; JsonObject other = (JsonObject) o; // size check if (this.size() != other.size()) return false; // value comparison for (String key : map.keySet()) { if (!other.containsKey(key)) { return false; } Object thisValue = this.getValue(key); Object otherValue = other.getValue(key); // identity check if (thisValue == otherValue) { continue; } // special case for numbers if (thisValue instanceof Number && otherValue instanceof Number && thisValue.getClass() != otherValue.getClass()) { Number n1 = (Number) thisValue; Number n2 = (Number) otherValue; // floating point values if (thisValue instanceof Float || thisValue instanceof Double || otherValue instanceof Float || otherValue instanceof Double) { // compare as floating point double if (n1.doubleValue() == n2.doubleValue()) { // same value check the next entry continue; } } if (thisValue instanceof Integer || thisValue instanceof Long || otherValue instanceof Integer || otherValue instanceof Long) { // compare as integer long if (n1.longValue() == n2.longValue()) { // same value check the next entry continue; } } } // special case for char sequences if (thisValue instanceof CharSequence && otherValue instanceof CharSequence && thisValue.getClass() != otherValue.getClass()) { CharSequence s1 = (CharSequence) thisValue; CharSequence s2 = (CharSequence) otherValue; if (Objects.equals(s1.toString(), s2.toString())) { // same value check the next entry continue; } } // fallback to standard object equals checks if (!Objects.equals(thisValue, otherValue)) { return false; } } // all checks passed return true; } @Override public int hashCode() { return map.hashCode(); } @Override public void writeToBuffer(Buffer buffer) { Buffer buf = toBuffer(); buffer.appendInt(buf.length()); buffer.appendBuffer(buf); } @Override public int readFromBuffer(int pos, Buffer buffer) { int length = buffer.getInt(pos); int start = pos + 4; Buffer buf = buffer.getBuffer(start, start + length); fromBuffer(buf); return pos + length + 4; } private void fromJson(String json) { map = Json.CODEC.fromString(json, Map.class); } private void fromBuffer(Buffer buf) { map = Json.CODEC.fromBuffer(buf, Map.class); } private static class Iter implements Iterator<Map.Entry<String, Object>> { final Iterator<Map.Entry<String, Object>> mapIter; Iter(Iterator<Map.Entry<String, Object>> mapIter) { this.mapIter = mapIter; } @Override public boolean hasNext() { return mapIter.hasNext(); } @Override public Map.Entry<String, Object> next() { final Map.Entry<String, Object> entry = mapIter.next(); final Object val = entry.getValue(); // perform wrapping final Object wrapped = wrapJsonValue(val); if (val != wrapped) { return new Entry(entry.getKey(), wrapped); } return entry; } @Override public void remove() { mapIter.remove(); } } private static final class Entry implements Map.Entry<String, Object> { final String key; final Object value; public Entry(String key, Object value) { this.key = key; this.value = value; } @Override public String getKey() { return key; } @Override public Object getValue() { return value; } @Override public Object setValue(Object value) { throw new UnsupportedOperationException(); } } static <T> Stream<T> asStream(Iterator<T> sourceIterator) { Iterable<T> iterable = () -> sourceIterator; return StreamSupport.stream(iterable.spliterator(), false); } }