浏览代码

提交接口调试内容

kaiweicai 3 年之前
父节点
当前提交
2698824e9f
共有 24 个文件被更改,包括 10795 次插入1 次删除
  1. 1 1
      build.gradle
  2. 0 0
      demo/src/main/assets/yyxx_game/yyxx_cfg.properties
  3. 0 0
      library_base/src/main/java/cn/yyxx/commsdk/base/YYXXApplication.java
  4. 11 0
      library_base/src/main/java/cn/yyxx/commsdk/base/YYXXWelcomeActivity.java
  5. 220 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/AESUtils.java
  6. 351 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/SignUtils.java
  7. 41 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/IGetCollection.java
  8. 550 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/JsonParser.java
  9. 465 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/JsonSerializer.java
  10. 55 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/MapNavigator.java
  11. 31 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/ParseException.java
  12. 16 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/SerializeException.java
  13. 157 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/TnJson.java
  14. 146 0
      library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/TnJsonBuilder.java
  15. 5 0
      library_core/src/main/cpp/comm_map.cpp
  16. 16 0
      library_core/src/main/cpp/comm_map.h
  17. 470 0
      library_core/src/main/cpp/external/json/json-forwards.h
  18. 2604 0
      library_core/src/main/cpp/external/json/json.h
  19. 5448 0
      library_core/src/main/cpp/external/json/jsoncpp.cpp
  20. 37 0
      library_core/src/main/cpp/yyxxgame.cpp
  21. 2 0
      library_core/src/main/java/cn/yyxx/commsdk/core/NTest.java
  22. 2 0
      library_core/src/main/java/cn/yyxx/commsdk/core/network/Hosts.java
  23. 2 0
      library_core/src/main/java/cn/yyxx/commsdk/core/network/SdkRequest.java
  24. 165 0
      library_core/src/main/java/cn/yyxx/commsdk/core/network/VolleyRequest.java

+ 1 - 1
build.gradle

@@ -11,7 +11,7 @@ buildscript {
         jcenter()
     }
     dependencies {
-        classpath 'com.android.tools.build:gradle:7.0.4'
+        classpath 'com.android.tools.build:gradle:7.1.1'
         classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
         // NOTE: Do not place your application dependencies here; they belong
         // in the individual module build.gradle files

+ 0 - 0
demo/src/main/assets/yyxx_game/yyxx_cfg.properties


+ 0 - 0
library_base/src/main/java/cn/yyxx/commsdk/YYXXApplication.java → library_base/src/main/java/cn/yyxx/commsdk/base/YYXXApplication.java


+ 11 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/YYXXWelcomeActivity.java

@@ -0,0 +1,11 @@
+package cn.yyxx.commsdk;
+
+import android.app.Activity;
+import android.os.Bundle;
+
+public class YYXXWelcomeActivity extends Activity {
+    @Override
+    protected void onCreate(Bundle savedInstanceState) {
+        super.onCreate(savedInstanceState);
+    }
+}

+ 220 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/AESUtils.java

@@ -0,0 +1,220 @@
+package com.yyrh.utils;
+
+import android.util.Log;
+
+import com.yyrh.utils.a2u.tn.utils.json.TnJson;
+
+import java.io.UnsupportedEncodingException;
+
+import java.security.InvalidAlgorithmParameterException;
+import java.security.InvalidKeyException;
+import java.security.NoSuchAlgorithmException;
+
+import java.util.HashMap;
+import java.util.Iterator;
+import java.util.Map;
+import java.util.Random;
+
+import javax.crypto.BadPaddingException;
+import javax.crypto.Cipher;
+import javax.crypto.IllegalBlockSizeException;
+import javax.crypto.NoSuchPaddingException;
+import javax.crypto.SecretKey;
+
+import javax.crypto.spec.IvParameterSpec;
+
+import javax.crypto.spec.SecretKeySpec;
+
+public class AESUtils {
+
+
+    private final int HASH_ITERATIONS = 10000;
+    private final int KEY_LENGTH = 256;
+    private char[] humanPassphrase = { 'P', 'e', 'r', ' ', 'v', 'a', 'l', 'l',
+            'u', 'm', ' ', 'd', 'u', 'c', 'e', 's', ' ', 'L', 'a', 'b', 'a',
+            'n', 't' };// per vallum duces labant
+    private byte[] salt = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0xA, 0xB, 0xC, 0xD,
+            0xE, 0xF }; // must save this for next time we want the key
+
+
+    private static final String CIPHERMODEPADDING = "AES/CBC/PKCS7Padding";// AES/CBC/PKCS7Padding
+
+
+    private static String ivParameter = "8eb99e1f9813f24b";// 密钥默认偏移,可更改
+
+
+    private static String sKey = "8eb99e1f9813f24b5edb7a501ceae5ac";// key必须为16位,可更改为自己的key
+
+
+
+    public static String encrypt(String param) {
+
+
+         SecretKeySpec skforAES = null;
+         byte[] iv = ivParameter.getBytes();
+         IvParameterSpec IV;
+
+
+        // This is our secret key. We could just save this to a file instead of
+        // regenerating it
+        // each time it is needed. But that file cannot be on the device (too
+        // insecure). It could
+        // be secure if we kept it on a server accessible through https.
+
+        // byte[] skAsByteArray = sk.getEncoded();
+        byte[] skAsByteArray;
+        try {
+            skAsByteArray = sKey.getBytes("ASCII");
+            skforAES = new SecretKeySpec(skAsByteArray, "AES");
+        } catch (UnsupportedEncodingException e) {
+            e.printStackTrace();
+        }
+        IV = new IvParameterSpec(iv);
+
+
+        byte[] plaintext  =  param.getBytes();
+        byte[] ciphertext = encrypt(CIPHERMODEPADDING, skforAES, IV, plaintext);
+        return bytesToHex(ciphertext);
+    }
+
+
+    /**
+     * 字节数组转16进制
+     * @param bytes 需要转换的byte数组
+     * @return  转换后的Hex字符串
+     */
+    public static String bytesToHex(byte[] bytes) {
+        StringBuffer sb = new StringBuffer();
+        for(int i = 0; i < bytes.length; i++) {
+            String hex = Integer.toHexString(bytes[i] & 0xFF);
+            if(hex.length() < 2){
+                sb.append(0);
+            }
+            sb.append(hex);
+        }
+        return sb.toString();
+    }
+
+    /**
+     * AES 加密
+     * @param cmp
+     * @param sk
+     * @param IV
+     * @param msg
+     * @return
+     */
+
+    private static byte[] encrypt(String cmp, SecretKey sk, IvParameterSpec IV,
+                           byte[] msg) {
+        try {
+            Cipher c = Cipher.getInstance(cmp);
+            c.init(Cipher.ENCRYPT_MODE, sk, IV);
+            return c.doFinal(msg);
+        } catch (NoSuchAlgorithmException nsae) {
+            Log.e("AESdemo", "no cipher getinstance support for " + cmp);
+        } catch (NoSuchPaddingException nspe) {
+            Log.e("AESdemo", "no cipher getinstance support for padding " + cmp);
+        } catch (InvalidKeyException e) {
+            Log.e("AESdemo", "invalid key exception");
+        } catch (InvalidAlgorithmParameterException e) {
+            Log.e("AESdemo", "invalid algorithm parameter exception");
+        } catch (IllegalBlockSizeException e) {
+            Log.e("AESdemo", "illegal block size exception");
+        } catch (BadPaddingException e) {
+            Log.e("AESdemo", "bad padding exception");
+        }
+        return null;
+    }
+
+
+
+
+    public static String rand(int digit) {
+        Random rand = new Random();
+        char[] letters = new char[]{'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q',
+                'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i',
+                'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'r',
+                '0', '1', '2', '3', '4', '5', '6', '7', '8', '9'};
+        String str = "";
+        int index;
+        boolean[] flags = new boolean[letters.length];//默认为false
+        for (int i = 0; i < digit; i++) {
+            do {
+                index = rand.nextInt(letters.length);
+            } while (flags[index] == true);
+            char c = letters[index];
+            str += c;
+            flags[index] = true;
+        }
+        return str;
+    }
+
+    /**
+     * hashMap转Json格式
+     * @param map
+     * @return
+     */
+    public static String hashMapToJson(HashMap<String, String> map) {
+        String string = "{";
+        for (Iterator it = map.entrySet().iterator(); it.hasNext();) {
+            Map.Entry e = (Map.Entry) it.next();
+            string += "\"" + e.getKey() + "\":";
+            String value = e.getValue() == null ? "" : e.getValue().toString();
+
+
+
+                string += "\"" + value + "\",";
+
+
+        }
+        string = string.substring(0, string.lastIndexOf(","));
+        string += "}";
+        return string;
+    }
+
+    public static String toJson(HashMap<String, String> map) {
+
+        String jsonString ="" ;//"[";
+
+        jsonString +=rand(5) + TnJson.toJson(map,TnJson.Mode.LIGHT);
+        YYLog.i(jsonString);
+        return jsonString ;//+= "]";
+
+    }
+
+
+
+    // 解密
+    public static String decrypt(String sSrc) {
+        try {
+            byte[] raw = sKey.getBytes("ASCII");
+            SecretKeySpec skeySpec = new SecretKeySpec(raw, "AES");
+            Cipher cipher = Cipher.getInstance(CIPHERMODEPADDING);
+            IvParameterSpec iv = new IvParameterSpec(ivParameter.getBytes());
+            cipher.init(Cipher.DECRYPT_MODE, skeySpec, iv);
+            byte[] encrypted1 = hexStringToByteArray(sSrc);// 先用base64解密
+            byte[] original = cipher.doFinal(encrypted1);
+            String originalString = new String(original, "utf-8");
+            return originalString;
+        } catch (Exception ex) {
+            return null;
+        }
+    }
+
+
+    /**
+     * 十六进制转换为字节数组
+     * @param s
+     * @return
+     */
+    public static byte[] hexStringToByteArray(String s) {
+        //十六进制转byte数组
+        int len = s.length();
+        byte[] bs = new byte[len/2];
+        for(int i = 0;i < len;i+=2) {
+            bs[i/2] = (byte) ((Character.digit(s.charAt(i), 16) << 4) + Character.digit(s.charAt(i+1), 16));
+        }
+        return bs;
+    }
+
+}

+ 351 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/SignUtils.java

@@ -0,0 +1,351 @@
+package com.yyrh.utils;
+
+import android.util.Base64;
+
+import java.io.ByteArrayOutputStream;
+import java.net.URLEncoder;
+import java.security.Key;
+import java.security.KeyFactory;
+import java.security.MessageDigest;
+import java.security.spec.X509EncodedKeySpec;
+import java.util.ArrayList;
+import java.util.Collections;
+
+import java.util.Comparator;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import javax.crypto.Cipher;
+import javax.crypto.Mac;
+import javax.crypto.spec.SecretKeySpec;
+
+import static com.yyrh.constant.URLConstants.APP_KEY;
+
+public class SignUtils {
+
+    private static final int MAX_ENCRYPT_BLOCK = 117;
+    public static final String KEY_ALGORITHM = "RSA";
+
+    public static String encryptParam(HashMap<String,String> map) {
+        String paramContent = "";
+        String URLenCodeparamContent = "";
+        List<Map.Entry<String, String>> infoIds = new ArrayList<Map.Entry<String, String>>(map.entrySet());
+        Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() {
+            @Override
+            public int compare(Map.Entry<String, String> o1, Map.Entry<String, String> o2) {
+                return o1.getKey().compareTo(o2.getKey());
+            }
+        });
+
+        for (int i = 0; i < infoIds.size(); i++) {
+
+            String id =  infoIds.get(i).toString();
+            paramContent = paramContent + id + "&";
+        }
+
+        String time = System.currentTimeMillis() / 1000 + "";
+        String allParam = paramContent + "time=" + time + "qDikI9v?>!x)1rS&yo";
+        String sign = Utils.getMD5(allParam);
+
+        for (String key : map.keySet()) {
+            YYLog.i(key + "=" + map.get(key));
+            URLenCodeparamContent = URLenCodeparamContent + key + "=" + toURLEncoded(map.get(key)) + "&";
+           }
+
+        URLenCodeparamContent =URLenCodeparamContent + "time=" + time  + "&sign="+ sign ;
+        return URLenCodeparamContent;
+    }
+
+    public static String paramSign(HashMap<String,String> map) {
+        String paramContent = "";
+        List<Map.Entry<String, String>> infoIds = new ArrayList<Map.Entry<String, String>>(map.entrySet());
+        Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() {
+            @Override
+            public int compare(Map.Entry<String, String> o1, Map.Entry<String, String> o2) {
+                return o1.getKey().compareTo(o2.getKey());
+            }
+        });
+
+        for (int i = 0; i < infoIds.size(); i++) {
+
+            String id =  infoIds.get(i).toString();
+            paramContent = paramContent + id + "&";
+        }
+
+        String time = toURLEncoded(System.currentTimeMillis() / 1000 + "");
+        paramContent = Utils.getMD5(paramContent + "time=" + time + "qDikI9v?>!x)1rS&yo");
+        return paramContent;
+    }
+
+    public static String encryptRsaByPublicKey(String key, String text)
+            throws Exception {
+
+        // 对公钥解密
+        byte[] data = Base64.decode(key, Base64.NO_WRAP);
+
+
+
+        byte[] byteText = text.getBytes();
+        // 取得公钥
+        X509EncodedKeySpec x509KeySpec = new X509EncodedKeySpec(data);
+        KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
+        Key publicKey = keyFactory.generatePublic(x509KeySpec);
+
+        // 对数据加密
+        Cipher cipher = Cipher.getInstance("RSA/ECB/PKCS1Padding");
+        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
+
+        int inputLen = byteText.length;
+        ByteArrayOutputStream out = new ByteArrayOutputStream();
+        int offSet = 0;
+        byte[] cache;
+        int i = 0;
+        while (inputLen -offSet > 0 ){
+            if(inputLen -offSet >MAX_ENCRYPT_BLOCK){
+                cache = cipher.doFinal(byteText,offSet,MAX_ENCRYPT_BLOCK);
+            }else{
+                cache = cipher.doFinal(byteText,offSet,inputLen - offSet);
+            }
+            out.write(cache,0,cache.length);
+            i++;
+            offSet = i * MAX_ENCRYPT_BLOCK;
+        }
+        byte[] encryptedData = out.toByteArray();
+        out.close();
+
+
+        String strBase64 = new String(Base64.encode(encryptedData, Base64.NO_WRAP));
+
+        return strBase64;
+    }
+
+    public static String toURLEncoded(String paramString) {
+        if (paramString == null || paramString.equals("")) {
+            return "";
+        }
+
+        try {
+            String str = new String(paramString.getBytes(), "UTF-8");
+            str = URLEncoder.encode(str, "UTF-8");
+            return str;
+        } catch (Exception localException) {
+            System.out.println("toURLEncoded error:" + paramString + ","
+                    + localException);
+        }
+
+        return "";
+    }
+
+
+
+    public static String paramSign(HashMap<String,String> map ,String time) {
+        String paramContent = "";
+        List<Map.Entry<String, String>> infoIds = new ArrayList<Map.Entry<String, String>>(map.entrySet());
+        Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() {
+            @Override
+            public int compare(Map.Entry<String, String> o1, Map.Entry<String, String> o2) {
+                return o1.getKey().compareTo(o2.getKey());
+            }
+        });
+
+        for (int i = 0; i < infoIds.size(); i++) {
+
+            String id =  infoIds.get(i).toString();
+            paramContent = paramContent + id + "&";
+
+
+
+        }
+        String content = paramContent +"time=" + time + "qDikI9v?>!x)1rS&yo";
+
+        paramContent = getMD5(content);
+        return paramContent;
+    }
+
+
+    public static String getMD5(String content) {
+        String s = null;
+        char hexDigits[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
+                'a', 'b', 'c', 'd', 'e', 'f'};
+        try {
+            java.security.MessageDigest md = java.security.MessageDigest
+                    .getInstance("MD5");
+            md.update(content.getBytes());
+            byte tmp[] = md.digest();
+            char str[] = new char[16 * 2];
+            int k = 0;
+            for (int i = 0; i < 16; i++) {
+                byte byte0 = tmp[i];
+                str[k++] = hexDigits[byte0 >>> 4 & 0xf];
+                str[k++] = hexDigits[byte0 & 0xf];
+            }
+            s = new String(str);
+
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+        return s;
+    }
+
+    /**
+     * 新融合增加打点上报签名规则
+     */
+
+    public static String sdkLogParamSign(HashMap<String,String> map) {
+        String paramContent = "";
+        List<Map.Entry<String, String>> infoIds = new ArrayList<Map.Entry<String, String>>(map.entrySet());
+        Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() {
+            @Override
+            public int compare(Map.Entry<String, String> o1, Map.Entry<String, String> o2) {
+                return o1.getKey().compareTo(o2.getKey());
+            }
+        });
+
+        for (int i = 0; i < infoIds.size(); i++) {
+
+            String id =  infoIds.get(i).toString();
+            paramContent = paramContent + id + "&";
+
+
+        }
+        String sign = "";
+        paramContent = paramContent.substring(0,paramContent.length() - 1);
+        try {
+            sign = hmacSHA256(paramContent,APP_KEY);
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+        return sign;
+    }
+
+    public static String hmacSHA256(String data, String key) throws Exception {
+
+        Mac sha256_HMAC = Mac.getInstance("HmacSHA256");
+
+        SecretKeySpec secret_key = new SecretKeySpec(key.getBytes("UTF-8"), "HmacSHA256");
+
+        sha256_HMAC.init(secret_key);
+
+        byte[] array = sha256_HMAC.doFinal(data.getBytes("UTF-8"));
+
+        StringBuilder sb = new StringBuilder();
+
+        for (byte item : array) {
+
+            sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));
+
+        }
+
+        return sb.toString();
+
+    }
+
+
+
+    public static String sdkLogEncryptParam(HashMap<String,String> map) {
+        String paramContent = "";
+        String URLenCodeparamContent = "";
+        List<Map.Entry<String, String>> infoIds = new ArrayList<Map.Entry<String, String>>(map.entrySet());
+        Collections.sort(infoIds, new Comparator<Map.Entry<String, String>>() {
+            @Override
+            public int compare(Map.Entry<String, String> o1, Map.Entry<String, String> o2) {
+                return o1.getKey().compareTo(o2.getKey());
+            }
+        });
+
+        for (int i = 0; i < infoIds.size(); i++) {
+
+            String id =  infoIds.get(i).toString();
+            paramContent = paramContent + id + "&";
+        }
+
+
+
+        String sign = sdkLogParamSign(map);
+
+        for (String key : map.keySet()) {
+            YYLog.i(key + "=" + map.get(key));
+            URLenCodeparamContent = URLenCodeparamContent + key + "=" + toURLEncoded(map.get(key)) + "&";
+        }
+
+        URLenCodeparamContent =URLenCodeparamContent  + "sign="+ sign ;
+        return URLenCodeparamContent;
+    }
+
+    /*
+     **************************************************** 上报log日志签名工具
+     */
+
+
+    /**
+     * 将byte转为16进制
+     *
+     * @param bytes
+     * @return
+     */
+    private static String byte2Hex(byte[] bytes) {
+        StringBuffer stringBuffer = new StringBuffer();
+        String temp = null;
+        for (int i = 0; i < bytes.length; i++) {
+            temp = Integer.toHexString(bytes[i] & 0xFF);
+            if (temp.length() == 1) {
+                //1得到一位的进行补0操作
+                stringBuffer.append("0");
+            }
+            stringBuffer.append(temp);
+        }
+        return stringBuffer.toString();
+    }
+
+
+
+
+
+
+    public static String sha256Hex(String s) throws Exception {
+        MessageDigest md = MessageDigest.getInstance("SHA-256");
+        byte[] d = md.digest(s.getBytes("UTF-8"));
+        return  byte2Hex(d).toLowerCase();
+    }
+
+
+    public static byte[] hmac256(byte[] key, String msg) throws Exception {
+        Mac mac = Mac.getInstance("HmacSHA256");
+        SecretKeySpec secretKeySpec = new SecretKeySpec(key, mac.getAlgorithm());
+        mac.init(secretKeySpec);
+        return mac.doFinal(msg.getBytes("UTF-8"));
+    }
+
+
+    /*
+     * 签名
+     */
+    public static String sdkLogParamSign(String time, String value) {
+        String sha256_value = null;
+        try {
+            sha256_value = sha256Hex(value);
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+
+
+        String sign = "timestamp=" + time + "\n" + sha256_value;
+
+        String hmac_sha256_value = "";
+        try {
+
+            hmac_sha256_value = byte2Hex(hmac256(APP_KEY.getBytes("UTF-8"), sign)).toLowerCase();;
+
+
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+
+        return hmac_sha256_value;
+    }
+
+
+
+}
+

+ 41 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/IGetCollection.java

@@ -0,0 +1,41 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+import java.util.Collection;
+import java.util.Map;
+
+/**
+ * The callback-interface for specifying type of collection which will be returned when parsing.
+ * @see TnJson#parse(String, IGetCollection)
+ */
+public interface IGetCollection {
+
+  /**
+   * Name of root element in a path. Using in call-back methods of IGetCollection.
+   * @see #forObject(String)
+   * @see #forList(String)
+   */
+  public static final String PATH_ROOT_KEY = JsonParser.PATH_ROOT_KEY;
+
+
+  /**
+   * This methow will be called when parcer need create new map.<br>
+   * If this method returns null, then map will be created with the default type - LinkedHashMap.
+   *
+   * @param path path of current element, starting from root. If this a root element, then path equal "root".
+   * @return the empty object who implementing interface from java.util.Map&lt;String, Object&gt;.
+   *
+   * @see #PATH_ROOT_KEY
+   */
+  Map<String, Object> forObject(String path);
+
+  /**
+   * This methow will be called when parcer need create new array.<br>
+   * If this method returns null, then array will be created with the default type - ArrayList.
+   *
+   * @param path path of current element, starting from root. If this a root element, then path equal "root".
+   * @return the empty object who implementing interface from java.util.Collection&lt;Object&gt;.
+   *
+   * @see #PATH_ROOT_KEY
+   */
+  Collection forList(String path);
+}

+ 550 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/JsonParser.java

@@ -0,0 +1,550 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+import java.math.BigInteger;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.LinkedHashMap;
+import java.util.Map;
+
+/**
+ * Parsing JSON string to Map object with JSON5
+ * JSON for Humans https://spec.json5.org
+ *
+ */
+class JsonParser {
+
+  /**
+   * This name will get element in a returned map when json-string will be an array of values.
+   */
+  public static final String DEFAULT_LIST_KEY = "list";
+
+  /**
+   * Name of root element in a path. Using in call-back methods of IGetCollection.
+   * @see IGetCollection#forObject(String)
+   * @see IGetCollection#forList(String)
+   */
+  public static final String PATH_ROOT_KEY = "root";
+
+
+  private IGetCollection listener;
+
+
+  /**
+   * Inner immutable class for represent path by root of json-object
+   */
+  private class Path {
+    private String path;
+
+    public Path(String str) {
+      this.path = str;
+    }
+
+    public String getName() {
+      return path;
+    }
+
+    public Path add(String node) {
+      return new Path(path + '.' + node);
+    }
+
+  }
+
+  private static final String NULL = "null";
+  private static final String BOOL_TRUE = "true";
+  private static final String BOOL_FALSE = "false";
+  private static final String NUM_INFINITY = "infinity";
+  private static final String NUM_INFINITY_PSITIVE = "+infinity";
+  private static final String NUM_INFINITY_NEGATIVE = "-infinity";
+  private static final String NUM_NAN = "nan";
+
+  private static final char LF = 0x0A;
+  private static final char CR = 0x0D;
+  private static final char LS = 0x2028;
+  private static final char PS = 0x2029;
+
+
+  private Map<String, Object> resultMap; // result object
+  private String content;                // incoming json-string
+  private int maxLength;                 // cache of length the incoming json-string
+  private int index;                     // current accepted symbol
+
+
+  JsonParser() {
+    //hide this
+  }
+  JsonParser(IGetCollection listener) {
+    this.listener = listener;
+  }
+
+
+
+  /**
+   * Prepare and start parsing
+   * @param data json-string
+   * @return java-map object - result of parsing
+   */
+  Map<String, Object> doParse(String data) {
+    content = data.trim();
+    maxLength = content.length();
+    index = 0;
+
+    Path emptypath = new Path(PATH_ROOT_KEY);
+    resultMap = getCollectionForObject(emptypath);
+
+    parseEmpty(emptypath);
+
+    return resultMap;
+  }
+
+  private void parseEmpty(Path emptypath) {
+    while (index < maxLength) {
+      char c = getTokenBegin();
+      if (c == '{') {
+        index++;
+        resultMap = parseMap(emptypath);
+        return;
+      }
+      else if (c == '[') {
+        index++;
+        Collection list = parseList(emptypath);
+        resultMap.put(DEFAULT_LIST_KEY, list);
+      }
+
+      index++;
+    }
+  }
+
+  private Map<String, Object> parseMap(Path path) {
+    Map<String, Object> map = getCollectionForObject(path);
+
+    while (index < maxLength) {
+      char c = getTokenBegin(); //skip to begin
+
+      if (c == '}') {
+        index++;
+        return map;
+      }
+
+      String key = extractIdenty(path);
+
+      String pathForLog = path.getName() + "." + key;
+
+      c = getTokenBegin();
+      if (c != ':') {
+        throw new ParseException("Invalid character '" + charToLog(c) + "' at position " + index + ", path '" + pathForLog + "', expected ':'.", index, pathForLog);
+      }
+      index++;
+      Object val = extractValue(path.add(key));
+      map.put(key, val);
+
+      c = getTokenBegin();
+      if (c == '}') {
+        index++;
+        return map;
+      }
+      if (c == ',') {
+        index++;
+        continue;
+      }
+
+      throw new ParseException("Invalid character '" + charToLog(c) + "' at position " + index + ", last path '"+ pathForLog +"', expected ',' or '}'.", index, pathForLog);
+    }
+
+    return map;
+  }
+
+  @SuppressWarnings("unchecked")
+  private Collection parseList(Path path) {
+    Collection list = getCollectionForList(path);
+
+    while (index < maxLength) {
+      char c = getTokenBegin();
+
+      if (c == ']') {
+        index++;
+        return list;
+      }
+
+      if (c == ',') {
+        index++;
+        continue;
+      }
+
+      Object val = extractValue(path);
+      list.add(val);
+
+    }
+
+    return list;
+  }
+
+
+  /**
+   * Return a first symbol which has mean, starting at position = index.
+   * Comments will be ignored.
+   * index will point to this symbol.
+   */
+  private char getTokenBegin() {
+    while (index < maxLength) {
+      char c = content.charAt(index);
+      if (Character.isLetterOrDigit(c) || c == '"' || c == '\'' || c == '@' || c == '#' || c == '$' || c == '_' || c == '{' || c == '}' || c == ':' || c == '[' || c == ']' || c == ',' || c == '+' || c == '-' || c == '.') {
+        return c;
+      }
+
+      if (c == '\\') {
+        return c;
+      }
+
+      if (c == '/') {
+        char next = content.charAt(index + 1);
+        if (next == '/') {
+          skipToEndLine();
+        }
+        else if (next == '*') {
+          index += 2;
+          skipToEndComent();
+        }
+      }
+
+      index++;
+    }
+    return 0;
+  }
+
+  private void skipToEndLine() {
+    while (index < maxLength) {
+      char c = content.charAt(index);
+      if (isLineTerminator(c)) {
+        return;
+      }
+      index++;
+    }
+  }
+
+  private void skipToEndComent() {
+    while (index < maxLength) {
+      char c = content.charAt(index);
+      if (c == '*') {
+        index++;
+        c = content.charAt(index);
+        if (c == '/') {
+          return;
+        }
+      }
+      index++;
+    }
+  }
+
+
+
+  private String extractIdenty(Path path) {
+    char c = content.charAt(index);
+    char terminator = (c == '"' || c == '\'') ? c : 0;
+
+    StringBuilder b = new StringBuilder();
+    while (index < maxLength) {
+      c = content.charAt(index);
+      if (c == terminator) {
+        index++;
+        if (b.length() == 0) {
+          continue;
+        }
+        return b.toString().trim();
+      }
+      if (terminator == 0 && (c == ':' || c == '/' || isWhiteSpace(c))) {
+        return b.toString().trim();
+      }
+
+      if (c  == '\\') {
+        c = getCharFromEscapedText();
+      }
+      if ((terminator == 0 && (c== '\'' || c == '"')) || isLineTerminator(c)) {
+        throw new ParseException("Invalid character '"+ charToLog(c) +"' for identifier '"+ b.toString() +"' at position "+ index +", path '"+ path.getName() +"'.", index, path.getName());
+      }
+      else {
+        b.append(c);
+      }
+      index++;
+    }
+
+    return b.toString().trim();
+  }
+
+  private Object extractValue(Path path) {
+    char c = getTokenBegin();
+
+    if (c == '{') {
+      index++;
+      Map<String, ?> map = parseMap(path);
+      return map;
+    }
+    if (c == '[') {
+      index++;
+      Collection list = parseList(path);
+      return list;
+    }
+    if (c == '"' || c == '\'') {
+      String str = extractString();
+      return str;
+    }
+    else {
+      Object num = extractLiteral(path);
+      return num;
+    }
+
+  }
+
+
+  private Object extractLiteral(Path path) {
+    StringBuilder b = new StringBuilder();
+    while (index < maxLength) {
+      char c = content.charAt(index);
+      if (c == '+' && b.length() == 0) {
+        index++;
+      }
+      else if (Character.isLetterOrDigit(c) || c == '.' || c == '+' || c == '-') {
+        b.append(c);
+        index++;
+      }
+      else {
+        break;
+      }
+    }
+
+    String literal = b.toString().trim().toLowerCase();
+
+    switch (literal) {
+      case NULL:
+        return null;
+      case BOOL_TRUE:
+        return true;
+      case BOOL_FALSE:
+        return false;
+      case NUM_INFINITY:
+        return Double.POSITIVE_INFINITY;
+      case NUM_INFINITY_PSITIVE:
+        return Double.POSITIVE_INFINITY;
+      case NUM_INFINITY_NEGATIVE:
+        return Double.NEGATIVE_INFINITY;
+      case NUM_NAN:
+        return Double.NaN;
+    }
+
+    try {
+      return detectNumber(literal);
+    }
+    catch (Exception ex) {
+      throw new ParseException("Invalid literal '" + literal + "' at position " + index + ", path '"+ path.getName() +"'.", index, path.getName());
+    }
+
+  }
+
+  private Object detectNumber(String literal) {
+    boolean hasDot = literal.indexOf('.') >= 0;
+    boolean hasE = literal.indexOf('e') >= 0;
+    boolean hasX = literal.indexOf('x') >= 0;
+
+    if (hasDot || (hasE && ! hasX)) {
+      return Double.parseDouble(literal);
+    }
+
+    // Integer.MAX_VALUE dec == 2147483647 - 10 characters
+    // Integer.MAX_VALUE hex == 0x7fffffff - 8 characters + 2 (0x) = 10 characters
+    // Long.MAX_VALUE dec == 9223372036854775807 - 19 characters
+    // Long.MAX_VALUE hex == 0x7fffffffffffffff - 16 characters + 2 (0x) = 18 characters
+    //
+    // therefore
+    // for maxintlen == 9 or 9 characters dec / hex
+    // for maxlonglen == 18 or 17 characters dec / hex
+    int maxintlen = hasX ? 9 : 9;
+    int maxlonglen = hasX ? 17 : 18;
+    if (literal.charAt(0) == '-') {
+      maxintlen++;
+      maxlonglen++;
+    }
+    if (literal.length() <= maxintlen) {
+      return Integer.decode(literal);
+    }
+    if (literal.length() <= maxlonglen) {
+      return Long.decode(literal);
+    }
+    else {
+      return new BigInteger(literal);
+    }
+  }
+
+
+  private String extractString() {
+    char terminator = content.charAt(index);
+    index++;
+
+    StringBuilder b = new StringBuilder();
+    while (index < maxLength) {
+      char c = content.charAt(index);
+      if (c == '\\') {
+        char ce = getCharFromEscapedText();
+        b.append(ce);
+        index++;
+      }
+      else if (c == terminator) {
+        index++;
+        break;
+      }
+      else {
+        b.append(c);
+        index++;
+      }
+    }
+
+    return b.toString().trim();
+  }
+
+  /**
+   * Extract escaped sequences, and convert to char
+   * https://spec.json5.org/#escapes
+   *
+   * As result index will refer to the last accepted character
+   *
+   * @return matching character
+   */
+  private char getCharFromEscapedText() {
+    char resultChar;
+    index++;
+    char next = content.charAt(index);
+    switch (next) {
+      case 'b': resultChar = '\b'; break;
+      case 'f': resultChar = '\f'; break;
+      case 'n': resultChar = '\n'; break;
+      case 'r': resultChar = '\r'; break;
+      case 't': resultChar = '\t'; break;
+      case 'v': resultChar = 0x000B; break;
+      case '0': resultChar = 0x0000; break;
+      case '\'': resultChar = '\'';  break;
+      case '"': resultChar = '"';  break;
+      case '\\': resultChar = '\\'; break;
+      case 'u':
+        StringBuilder bu = new StringBuilder();
+        bu.append(content.charAt(++index));
+        bu.append(content.charAt(++index));
+        bu.append(content.charAt(++index));
+        bu.append(content.charAt(++index));
+        int hexValU = Integer.parseInt(bu.toString(), 16);
+        resultChar = (char) hexValU;
+        break;
+
+      case 'x':
+      case 'X':
+        StringBuilder bx = new StringBuilder();
+        char cx = content.charAt(index + 1); //index will refer to the last accepted character, therefore we need to use +1
+        while (isHexadecimalChar(cx)) {
+          bx.append(cx);
+          index++;
+          cx = content.charAt(index + 1);
+        }
+        int hexValX = Integer.parseInt(bx.toString(), 16);
+        resultChar = (char) hexValX;
+        break;
+
+      default: return next;
+
+    }
+
+    return resultChar;
+  }
+
+
+  /**
+   * Line terminator point out to end of single-line comment
+   */
+  private boolean isLineTerminator(char c) {
+    return c == LF || c == CR || c == LS || c == PS;
+  }
+  private boolean isWhiteSpace(char c) {
+    switch (c) {
+      case 0x0009:
+      case 0x000A:
+      case 0x000B:
+      case 0x000C:
+      case 0x000D:
+      case 0x0020:
+      case 0x0085:
+      case 0x00A0:
+      case 0x1680:
+      case 0x2000:
+      case 0x2001:
+      case 0x2002:
+      case 0x2003:
+      case 0x2004:
+      case 0x2005:
+      case 0x2006:
+      case 0x2007:
+      case 0x2008:
+      case 0x2009:
+      case 0x200A:
+      case 0x2028:
+      case 0x2029:
+      case 0x202F:
+      case 0x205F:
+      case 0x3000:
+        return true;
+      default: return false;
+    }
+  }
+  private boolean isHexadecimalChar(char c) {
+    if (c >= '0' && c <= '9') {
+      return true;
+    }
+    if (c >= 'A' && c <= 'F') {
+      return true;
+    }
+    if (c >= 'a' && c <= 'f') {
+      return true;
+    }
+    return false;
+  }
+
+  /**
+   * Transform character to string for log
+   */
+  private String charToLog(char c) {
+    switch (c) {
+      case '\b': return "\\b";
+      case '\f': return "\\f";
+      case '\n': return "\\n";
+      case '\r': return "\\r";
+      case '\t': return "\\t";
+      case '\'': return "'";
+      case '"': return "\"";
+      case LS: return "LS (0x2028)";
+      case PS: return "PS (0x2029)";
+
+      case 0: return "null (0x0000)";
+
+      default: return String.valueOf(c);
+    }
+
+  }
+
+  private Map<String, Object> getCollectionForObject(Path path) {
+    Map<String, Object> result = null;
+    if (listener != null) {
+      result = listener.forObject(path.getName());
+    }
+    if (result == null) {
+      result = new LinkedHashMap<>();
+    }
+    return result;
+  }
+
+  private Collection getCollectionForList(Path path) {
+    Collection result = null;
+    if (listener != null) {
+      result = listener.forList(path.getName());
+    }
+    if (result == null) {
+      result = new ArrayList();
+    }
+    return result;
+  }
+
+}

+ 465 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/JsonSerializer.java

@@ -0,0 +1,465 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+import java.lang.reflect.Array;
+import java.lang.reflect.Field;
+import java.lang.reflect.InvocationTargetException;
+import java.lang.reflect.Method;
+import java.lang.reflect.Modifier;
+import java.time.LocalDate;
+import java.time.LocalDateTime;
+import java.time.LocalTime;
+import java.time.format.DateTimeFormatter;
+import java.util.Collection;
+import java.util.Date;
+import java.util.Map;
+
+/**
+ * MAP to JSON converter.<br>
+ * Every value in the Map must be
+ * <ul>
+ *   <li>either a simple value: string, number, boolean, or other;</li>
+ *   <li>either by a Map such as Map&lt;String, Object&gt;;</li>
+ *   <li>either an array of values such as Collection&lt;Object&gt;.</li>
+ * </ul>
+ * <br>
+ * Of course, you can use for json-object any class inheritor of the java.util.Map interface.<br>
+ * For json-array you can use any class inheritor of the java.util.Collection interface.
+ */
+class JsonSerializer {
+
+  private JsonSerializer() {
+    //hide this
+  }
+
+  /**
+   * Convert Map to JSON with builder
+   * @param data Object to convert. It can be Map, Collection, array, or any other object
+   * @param builder json generator tuning
+   * @return JSON-string
+   * @see TnJsonBuilder
+   */
+  public static String toJson(Object data, TnJsonBuilder builder) {
+    StringBuilder b = new StringBuilder();
+    addValue(data, b, builder, 0, "");
+    String json = b.toString();
+    return json;
+  }
+
+  private static void addValue(Object value, StringBuilder b, TnJsonBuilder builder, int level, String path) {
+
+    if (value == null) {
+      b.append("null");
+    }
+    else if (value instanceof Character) {
+      addString(String.valueOf(value), b, builder);
+    }
+    else if (value instanceof CharSequence) {
+      addString((CharSequence) value, b, builder);
+    }
+    else if (value instanceof Number) {
+      addNum(value, b);
+    }
+    else if (value instanceof Boolean) {
+      boolean v = (Boolean) value;
+      b.append(v ? "true" : "false");
+    }
+    else if (value instanceof Date) {
+      long v = ((Date)value).getTime();
+      b.append(v);
+    }
+
+    /**
+     * 由于 LocalDate LocalTime LocalDateTime 只能在8.0上运行,
+     * 且解析内容用不上,因此注释相关代码块,避免兼容问题
+     *
+    else if (value instanceof LocalDate) {
+      String strd = ((LocalDate)value).format(DateTimeFormatter.ISO_DATE);
+      b.append(strd);
+    }
+    else if (value instanceof LocalTime) {
+      String strt = ((LocalTime)value).format(DateTimeFormatter.ISO_TIME);
+      b.append(strt);
+    }
+    else if (value instanceof LocalDateTime) {
+      String strdt = ((LocalDateTime)value).format(DateTimeFormatter.ISO_DATE_TIME);
+      b.append(strdt);
+    }
+     */
+    else if (value instanceof Map) {
+      addMap((Map) value, b, builder, level, path);
+    }
+    else if (value instanceof Collection) {
+      addList((Collection) value, b, builder, level, path);
+    }
+    else if (value.getClass().isArray()) {
+      addArray(value, b, builder, level, path);
+    }
+    else {
+      addObj(value, b, builder, level, path);
+    }
+
+  }
+
+  private static void addString(CharSequence str, StringBuilder b, TnJsonBuilder builder) {
+    if (str == null) {
+      b.append("null");
+    }
+    else {
+      b.append(builder.quoteSymbol);
+
+      int len = str.length();
+      for (int p = 0; p < len; p++) {
+        char c = str.charAt(p);
+        if (Character.isLetterOrDigit(c)) {
+          b.append(c);
+        }
+        else if (c == '\n') {
+          if (builder.isAllowMultiRowString) {
+            b.append("/\n");
+          }
+          else if (builder.isStayReadable) {
+            b.append(charToReadable(c));
+          }
+          else {
+            b.append(c);
+          }
+        }
+        else if (c == builder.quoteSymbol) {
+          if (builder.isStayReadable) {
+            b.append(charToReadable(c));
+          }
+          else {
+            b.append(c);
+          }
+        }
+        else {
+          if (builder.isStayReadable) {
+            if (c == '\'' || c == '"') {
+              b.append(c);
+            }
+            else {
+              b.append(charToReadable(c));
+            }
+          }
+          else {
+            b.append(c);
+          }
+        }
+      }
+
+      b.append(builder.quoteSymbol);
+    }
+  }
+  private static String charToReadable(char c) {
+    switch (c) {
+      case ' ': return String.valueOf(c);
+      case '`': return String.valueOf(c);
+      case '~': return String.valueOf(c);
+      case '!': return String.valueOf(c);
+      case '@': return String.valueOf(c);
+      case '#': return String.valueOf(c);
+      case '$': return String.valueOf(c);
+      case '%': return String.valueOf(c);
+      case '^': return String.valueOf(c);
+      case '&': return String.valueOf(c);
+      case '*': return String.valueOf(c);
+      case '(': return String.valueOf(c);
+      case ')': return String.valueOf(c);
+      case '-': return String.valueOf(c);
+      case '_': return String.valueOf(c);
+      case '=': return String.valueOf(c);
+      case '+': return String.valueOf(c);
+      case '[': return String.valueOf(c);
+      case ']': return String.valueOf(c);
+      case '{': return String.valueOf(c);
+      case '}': return String.valueOf(c);
+      case ';': return String.valueOf(c);
+      case ':': return String.valueOf(c);
+      case '"': return "\\\"";
+      case '\'': return "\\'";
+      case '\\': return "\\\\";
+      case '|': return String.valueOf(c);
+      case '/': return "\\/";
+      case ',': return String.valueOf(c);
+      case '.': return String.valueOf(c);
+      case '?': return String.valueOf(c);
+      case '<': return String.valueOf(c);
+      case '>': return String.valueOf(c);
+      case '\b': return "\\b";
+      case '\f': return "\\f";
+      case '\n': return "\\n";
+      case '\r': return "\\r";
+      case '\t': return "\\t";
+      default: return unicodeEscaped(c);
+    }
+  }
+  private static String unicodeEscaped(char ch) {
+    if (ch < 0x10) {
+      return "\\u000" + Integer.toHexString(ch);
+    }
+    else if (ch < 0x100) {
+      return "\\u00" + Integer.toHexString(ch);
+    }
+    else if (ch < 0x1000) {
+      return "\\u0" + Integer.toHexString(ch);
+    }
+    return "\\u" + Integer.toHexString(ch);
+  }
+
+
+  private static void addNum(Object num, StringBuilder b) {
+    String v  = String.valueOf(num);
+    b.append(v);
+  }
+
+  private static void addMap(Map map, StringBuilder b, TnJsonBuilder builder, int level, String path) {
+    int valuelevel = level + 1;
+
+    b.append("{");
+    endLine(b, builder);
+
+    boolean hasEntry = false;
+    for(Object keyObj : map.keySet()) {
+      String key = String.valueOf(keyObj);
+      Object value = map.get(keyObj);
+      String valuePath = path+"."+key;
+
+      if (builder.pathHandler != null) {
+        value = builder.pathHandler.handlePath(valuePath, value);
+      }
+      if (value == null && !builder.isKeepNull) {
+        continue;
+      }
+      if (value != null && builder.typeHandler != null) {
+        value = builder.typeHandler.handleType(value);
+      }
+
+      key = codeKey(key, builder);
+
+      if (hasEntry) {
+        b.append(",");
+        endLine(b, builder);
+      }
+      else {
+        hasEntry = true;
+      }
+
+      startLine(b, builder, valuelevel);
+      b.append(key);
+
+      b.append(":");
+      if (builder.isFormated) {
+        b.append(" ");
+      }
+
+      addValue(value, b, builder, valuelevel, valuePath);
+    }
+
+    endLine(b, builder);
+    startLine(b, builder, level);
+    b.append("}");
+
+  }
+
+  private static String codeKey(CharSequence key, TnJsonBuilder builder) {
+    StringBuilder b = new StringBuilder();
+    int len = key.length();
+    boolean validIdenty = true;
+    for (int i=0; i < len; i++) {
+      char c = key.charAt(i);
+      if (Character.isLetterOrDigit(c)) {
+        b.append(c);
+      }
+      else {
+        b.append(c);
+        validIdenty = false;
+      }
+
+
+    }
+
+    if (!builder.isKeyQuoted && validIdenty && Character.isLetter(key.charAt(0))) {
+      return b.toString();
+    }
+    else {
+      return builder.quoteSymbol + b.toString() + builder.quoteSymbol;
+    }
+  }
+
+  private static void addList(Collection list, StringBuilder b, TnJsonBuilder builder, int level, String path) {
+    int itemlevel = level + 1;
+
+    b.append("[");
+
+    endLine(b, builder);
+    startLine(b, builder, itemlevel);
+
+    boolean hasEntry = false;
+    for (Object value : list) {
+
+      if (builder.pathHandler != null) {
+        value = builder.pathHandler.handlePath(path, value);
+      }
+      if (value == null && !builder.isKeepNull) {
+        continue;
+      }
+
+      if (hasEntry) {
+        b.append(",");
+        endLine(b, builder);
+        startLine(b, builder, itemlevel);
+      }
+      else {
+        hasEntry = true;
+      }
+
+
+      if (value != null && builder.typeHandler != null) {
+        value = builder.typeHandler.handleType(value);
+      }
+
+      addValue(value, b, builder, itemlevel, path);
+    }
+    endLine(b, builder);
+    startLine(b, builder, level);
+    b.append("]");
+
+  }
+
+  private static void addArray(Object array, StringBuilder b, TnJsonBuilder builder, int level, String path) {
+    int itemlevel = level + 1;
+
+    b.append("[");
+
+    endLine(b, builder);
+    startLine(b, builder, itemlevel);
+
+    boolean hasEntry = false;
+    int length = Array.getLength(array);
+    for (int i = 0; i < length; i ++) {
+      Object value = Array.get(array, i);
+      if (builder.pathHandler != null) {
+        value = builder.pathHandler.handlePath(path, value);
+      }
+      if (value == null && !builder.isKeepNull) {
+        continue;
+      }
+
+      if (hasEntry) {
+        b.append(",");
+        endLine(b, builder);
+        startLine(b, builder, itemlevel);
+      }
+      else {
+        hasEntry = true;
+      }
+
+      if (value != null && builder.typeHandler != null) {
+        value = builder.typeHandler.handleType(value);
+      }
+
+      addValue(value, b, builder, itemlevel, path);
+    }
+
+    endLine(b, builder);
+    startLine(b, builder, level);
+    b.append("]");
+
+  }
+
+  private static void addObj(Object obj, StringBuilder b, TnJsonBuilder builder, int level, String path) {
+
+    Class<?> cls = obj.getClass();
+
+    try {
+      Method toJsonMtd = cls.getDeclaredMethod("toJson");
+      toJsonMtd.setAccessible(true);
+      String string = (String) toJsonMtd.invoke(obj);
+      b.append(string);
+      return;
+    }
+    catch (NoSuchMethodException e) {
+      //no method, no problem
+    }
+    catch (InvocationTargetException | IllegalAccessException e) {
+      throw new SerializeException("Error on invoke method toJson by class: "+cls.getName()+", object: " + String.valueOf(obj)+".", e);
+    }
+    int valuelevel = level + 1;
+
+    b.append("{");
+    endLine(b, builder);
+
+    boolean hasEntry = false;
+    Field[] fields = cls.getDeclaredFields();
+    for (Field field : fields) {
+      int modifiers = field.getModifiers();
+      boolean allow = !field.isSynthetic() && !Modifier.isPrivate(modifiers) && !Modifier.isTransient(modifiers);
+      if (! allow) {
+        continue;
+      }
+
+      Object value = getObjValue(field, obj);
+
+      if (value == null && !builder.isKeepNull) {
+        continue;
+      }
+      if (value != null && builder.typeHandler != null) {
+        value = builder.typeHandler.handleType(value);
+      }
+
+      if (hasEntry) {
+        b.append(",");
+        endLine(b, builder);
+      }
+      else {
+        hasEntry = true;
+      }
+
+      startLine(b, builder, valuelevel);
+
+      b.append(field.getName());
+      b.append(":");
+
+      if (builder.isFormated) {
+        b.append(" ");
+      }
+
+      addValue(value, b, builder, level, path);
+    }
+    endLine(b, builder);
+    startLine(b, builder, level);
+    b.append("}");
+  }
+  private static Object getObjValue(Field field, Object fromObj) {
+    boolean accessible = field.isAccessible();
+    field.setAccessible(true);
+    try {
+      Object value = field.get(fromObj);
+      return value;
+    }
+    catch (Throwable t) {
+      throw new SerializeException("Error on extract value from object " + fromObj.getClass().getName() + " ("+String.valueOf(fromObj) + ") from field " + field.getName(), t);
+    }
+    finally {
+      field.setAccessible(accessible);
+    }
+  }
+
+
+
+  private static void endLine(StringBuilder b, TnJsonBuilder builder) {
+    if (builder.isFormated) {
+      b.append("\n");
+    }
+  }
+  private static void startLine(StringBuilder b, TnJsonBuilder builder, int level) {
+    if (!builder.isFormated) {
+      return;
+    }
+
+    for(int i=0; i<level; i++) {
+      b.append("  ");
+    }
+  }
+
+}

+ 55 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/MapNavigator.java

@@ -0,0 +1,55 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+import java.util.Arrays;
+import java.util.List;
+import java.util.Map;
+
+/**
+ * Navigate by Map represented object.<br>
+ * Every value in the Map must be
+ * <ul>
+ *   <li>either a simple value: string, number, boolean, or other;</li>
+ *   <li>either by a Map such as Map&lt;String, Object&gt;;</li>
+ *   <li>either an array of values such as Collection&lt;Object&gt;.</li>
+ * </ul>
+ */
+public class MapNavigator {
+
+  private MapNavigator() {
+    //hide this
+  }
+
+
+  /**
+   * Find value from map by path.
+   * @param map map with values
+   * @param path path with string-keys and dot as separator
+   * @return value
+   */
+  public static Object fromPath(Map map, String path) {
+
+    Object res = map;
+    List<String> pathList =  Arrays.asList(path.split("\\."));
+
+    StringBuilder node = new StringBuilder();
+
+    for (String pat : pathList) {
+      if (res == null) {
+        return null;
+      }
+      if (res instanceof Map) {
+        res = ((Map) res).get(pat);
+      }
+      else if (res instanceof List) {
+        int ix = Integer.parseInt(pat);
+        res = ((List) res).get(ix);
+      }
+      else {
+        throw new RuntimeException("Incorrect path: node=" + node.toString() + " is not a Map or List.");
+      }
+      node.append(".").append(pat);
+    }
+
+    return res;
+  }
+}

+ 31 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/ParseException.java

@@ -0,0 +1,31 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+/**
+ * Exception on parse error
+ */
+public class ParseException extends RuntimeException {
+
+  private int position;
+  private String path;
+
+  public ParseException(String s, int position, String path) {
+    super(s);
+    this.position = position;
+    this.path = path;
+  }
+
+  /**
+   * Position in json where occur error.
+   * @return position of invalid symbol
+   */
+  public int getPosition() {
+    return position;
+  }
+  /**
+   * Path in json where occur error.
+   * @return path where occur error
+   */
+  public String getPath() {
+    return path;
+  }
+}

+ 16 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/SerializeException.java

@@ -0,0 +1,16 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+/**
+ * Exception on serialization error
+ */
+public class SerializeException extends RuntimeException {
+
+  public SerializeException(String s) {
+    super(s);
+  }
+
+  public SerializeException(String s, Throwable cause) {
+    super(s, cause);
+  }
+
+}

+ 157 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/TnJson.java

@@ -0,0 +1,157 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+import java.util.Map;
+
+/**
+ * Using face
+ */
+public class TnJson {
+
+  private static final TnJsonBuilder SET_HARD         = TnJsonBuilder.init();
+  private static final TnJsonBuilder SET_LIGHT        = TnJsonBuilder.init().readable();
+  private static final TnJsonBuilder SET_FORMATTED    = TnJsonBuilder.init().readable().formated();
+  private static final TnJsonBuilder SET_JSON5        = TnJsonBuilder.init().readable().formated().withoutKeyQuote().allowMultiRowString();
+  private static final TnJsonBuilder SET_JSON5COMPACT = TnJsonBuilder.init().readable().withoutKeyQuote().singleQuote();
+
+  /**
+   * This name will get element in a returned map when json-string will be an array of values.
+   */
+  public static final String DEFAULT_LIST_KEY = JsonParser.DEFAULT_LIST_KEY;
+
+
+  /**
+   * JSON generation mode
+   * @see #toJson(Object, Mode)
+   */
+  public enum Mode {
+    /**
+     * Will be generated compact json-string, where any non-digital and non-letter character in string will be replaced with sequence uXXXX.<br>
+     * This mode is default, because it has max compatibility with other clients.
+     */
+    HARD,
+
+    /**
+     * Will be generated compact json-string, where non-digital and non-letter character in string will be stay in readable format, if it possible.<bR>
+     * This format is more compact, but is not all client can parse it.
+     */
+    LIGHT,
+
+    /**
+     * Will be generated json-string in pretty read format, where non-digital and non-letter character in string will be stay in readable format, if it possible.<bR>
+     *
+     */
+    FORMATTED,
+
+    /**
+     * Will be generated json-string in max human readable format json5.<br>
+     * See detail about json5 on https://json5.org/
+     */
+    JSON5,
+
+    /**
+     * JSON5 like, but without linefeed
+     */
+    JSON5COMPACT
+  }
+
+  public static TnJsonBuilder builder() {
+    return new TnJsonBuilder();
+  }
+
+
+  /**
+   * Convert Object to JSON.<br>
+   * Will be generated compact json-string, where any non-digital and non-letter character in string will be replaced with sequence uXXXX.
+   * @param data Object to convert it can be Map, Collection, array, or any other object
+   * @return JSON-string
+   */
+  public static String toJson(Object data) {
+    return SET_HARD.buildJson(data);
+  }
+
+  /**
+   * Convert Map to JSON with specify output string format
+   * @param data Object to convert. It can be Map, Collection, array, or any other object
+   * @param mode affects the format of resulting string
+   * @return JSON-string
+   * @see Mode
+   */
+  public static String toJson(Object data, Mode mode) {
+    switch (mode) {
+      case HARD:         return SET_HARD.buildJson(data);
+      case LIGHT:        return SET_LIGHT.buildJson(data);
+      case FORMATTED:    return SET_FORMATTED.buildJson(data);
+      case JSON5:        return SET_JSON5.buildJson(data);
+      case JSON5COMPACT: return SET_JSON5COMPACT.buildJson(data);
+      default:           return SET_HARD.buildJson(data);
+    }
+  }
+
+
+  /**
+   * Parsing JSON-string to Map.<br>
+   * Every value in the resulting Map will be:
+   * <ul>
+   *   <li>or a simple value (string or number or boolean),</li>
+   *   <li>or a LinkedHashMap with nested json-object,</li>
+   *   <li>or an ArrayList of values.</li>
+   * </ul>
+   *
+   * @param data incoming JSON-string.
+   * @return Map with data.<br>
+   * If JSON contain only array, such as [1,2] then will return Map
+   * with single element by key-name DEFAULT_ROOT, which contain list.
+   * @see #DEFAULT_LIST_KEY
+   *
+   */
+  public static Map<String, Object> parse(String data) {
+    try {
+      JsonParser p = new JsonParser();
+      return p.doParse(data);
+    }
+    catch (ParseException px) {
+      //Hide unnecessary log trace. If you want full trace - change it.
+      throw new ParseException(px.getMessage(), px.getPosition(), px.getPath());
+    }
+  }
+
+  /**
+   * Parsing JSON-string to Map with specifying returned collections.<br>
+   * For each element representing a non-simple value, will be called the corresponding method of listener,
+   * which allows you to set type of the returned object.
+   * @see IGetCollection
+   *
+   * @param data incoming JSON-string.
+   * @param listener callback listener.
+   * @return Map with data.<br>
+   * If JSON contain only array, such as [1,2] then will return Map
+   * with single element by key-name DEFAULT_LIST_KEY, which contain list.
+   * @see JsonParser#DEFAULT_LIST_KEY
+   */
+  public static Map<String, Object> parse(String data, IGetCollection listener) {
+    try {
+      JsonParser p = new JsonParser(listener);
+      return p.doParse(data);
+    }
+    catch (ParseException px) {
+      //Hide unnecessary log trace. If you want full trace - change it.
+      throw new ParseException(px.getMessage(), px.getPosition(), px.getPath());
+    }
+  }
+
+  TnJson() {
+    //hide constructor
+  }
+
+
+  /**
+   * Build JSON string. As default this method equivalent with static method toJson(), but this method using TnJsonBuilder
+   * @param data Object to convert. It can be Map, Collection, array, or any other object
+   * @return JSON-string
+   */
+  public String buildJson(Object data) {
+    return SET_HARD.buildJson(data);
+  }
+
+
+}

+ 146 - 0
library_base/src/main/java/cn/yyxx/commsdk/base/utils/sign/a2u/tn/utils/json/TnJsonBuilder.java

@@ -0,0 +1,146 @@
+package com.yyrh.utils.a2u.tn.utils.json;
+
+/**
+ * json generation configurator
+ */
+public class TnJsonBuilder extends TnJson {
+  boolean isKeyQuoted = true;
+  char quoteSymbol = '"';
+  boolean isFormated = false;
+  boolean isStayReadable = false;
+  boolean isAllowMultiRowString = false;
+  boolean isKeepNull = false;
+
+
+  /**
+   * Interfase for handing generating json in concrete path
+   */
+  public interface IPathHandler {
+    /**
+     * This methow will be called when json generator start serialize new value.<br>
+     * @param path path from root element witn dot as separator sample ".num", ".internal.cls"
+     * @param value value in this path
+     * @return the json-string or other value which be converted to json by standard rules
+     */
+    Object handlePath(String path, Object value);
+  }
+
+  IPathHandler pathHandler = null;
+
+
+  /**
+   * Interface for handing generating json by concrete type
+   */
+  public interface ITypeHandler {
+    /**
+     * This methow will be called when json generator start serialize new value.<br>
+     * @param value not null value for serialization
+     * @return the json-string or other value which be converted to json by standard rules
+     */
+    Object handleType(Object value);
+  }
+
+  ITypeHandler typeHandler = null;
+
+
+
+  TnJsonBuilder() {
+    //hide constructor
+  }
+
+
+
+  /**
+   * Create new builder
+   * @return new object with default settings
+   */
+  static TnJsonBuilder init() {
+    return new TnJsonBuilder();
+  }
+
+  /**
+   * Disable generating quotes in the key
+   * @return this builder
+   */
+  public TnJsonBuilder withoutKeyQuote() {
+    isKeyQuoted = false;
+    return this;
+  }
+
+  /**
+   * Use single quotes
+   * @return this builder
+   */
+  public TnJsonBuilder singleQuote() {
+    quoteSymbol = '\'';
+    return this;
+  }
+
+  /**
+   * Format the final json
+   * @return this builder
+   */
+  public TnJsonBuilder formated() {
+    isFormated = true;
+    return this;
+  }
+
+  /**
+   * Leave characters in a strings as readable as possible
+   * @return this builder
+   */
+  public TnJsonBuilder readable() {
+    isStayReadable = true;
+    return this;
+  }
+
+  /**
+   * Allow linefeed in a strings
+   * @return this builder
+   */
+  public TnJsonBuilder allowMultiRowString() {
+    isAllowMultiRowString = true;
+    return this;
+  }
+
+  /**
+   * Allow null values
+   * @return this builder
+   */
+  public TnJsonBuilder keepNull() {
+    isKeepNull = true;
+    return this;
+  }
+
+  /**
+   * Set handler for generating json in concrete path
+   * @param pathHandler handler @see IPathHandler
+   * @return this builder
+   */
+  public TnJsonBuilder handlePath(IPathHandler pathHandler) {
+    this.pathHandler = pathHandler;
+    return this;
+  }
+
+  /**
+   * Set handler for generating json by concrete type
+   * @param typeHandler handler @see ITypeHandler
+   * @return this builder
+   */
+  public TnJsonBuilder handleType(ITypeHandler typeHandler) {
+    this.typeHandler = typeHandler;
+    return this;
+  }
+
+
+  /**
+   * Building JSON by parameters in this builder
+   * @param data Object to convert. It can be Map, Collection, array, or any other object
+   * @return JSON-string
+   */
+  @Override
+  public String buildJson(Object data) {
+    return JsonSerializer.toJson(data, this);
+  }
+
+}

+ 5 - 0
library_core/src/main/cpp/comm_map.cpp

@@ -0,0 +1,5 @@
+//
+// Created by KaiWeiCai on 2022/1/27.
+//
+
+#include "comm_map.h"

+ 16 - 0
library_core/src/main/cpp/comm_map.h

@@ -0,0 +1,16 @@
+//
+// Created by KaiWeiCai on 2022/1/27.
+//
+
+#ifndef YYXXCOMMSDK_COMM_MAP_H
+#define YYXXCOMMSDK_COMM_MAP_H
+
+
+
+class comm_map {
+
+};
+
+
+
+#endif //YYXXCOMMSDK_COMM_MAP_H

+ 470 - 0
library_core/src/main/cpp/external/json/json-forwards.h

@@ -0,0 +1,470 @@
+/// Json-cpp amalgamated forward header (http://jsoncpp.sourceforge.net/).
+/// It is intended to be used with #include "json/json-forwards.h"
+/// This header provides forward declaration for all JsonCpp types.
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: LICENSE
+// //////////////////////////////////////////////////////////////////////
+
+/*
+The JsonCpp library's source code, including accompanying documentation, 
+tests and demonstration applications, are licensed under the following
+conditions...
+
+Baptiste Lepilleur and The JsonCpp Authors explicitly disclaim copyright in all 
+jurisdictions which recognize such a disclaimer. In such jurisdictions, 
+this software is released into the Public Domain.
+
+In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
+2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur and
+The JsonCpp Authors, and is released under the terms of the MIT License (see below).
+
+In jurisdictions which recognize Public Domain property, the user of this 
+software may choose to accept it either as 1) Public Domain, 2) under the 
+conditions of the MIT License (see below), or 3) under the terms of dual 
+Public Domain/MIT License conditions described here, as they choose.
+
+The MIT License is about as close to Public Domain as a license can get, and is
+described in clear, concise terms at:
+
+   http://en.wikipedia.org/wiki/MIT_License
+   
+The full text of the MIT License follows:
+
+========================================================================
+Copyright (c) 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+
+Permission is hereby granted, free of charge, to any person
+obtaining a copy of this software and associated documentation
+files (the "Software"), to deal in the Software without
+restriction, including without limitation the rights to use, copy,
+modify, merge, publish, distribute, sublicense, and/or sell copies
+of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
+BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
+ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
+========================================================================
+(END LICENSE TEXT)
+
+The MIT license is compatible with both the GPL and commercial
+software, affording one all of the rights of Public Domain with the
+minor nuisance of being required to keep the above copyright notice
+and license text in the source code. Note also that by accepting the
+Public Domain "license" you can re-license your copy using whatever
+license you like.
+
+*/
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: LICENSE
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+#ifndef JSON_FORWARD_AMALGAMATED_H_INCLUDED
+# define JSON_FORWARD_AMALGAMATED_H_INCLUDED
+/// If defined, indicates that the source file is amalgamated
+/// to prevent private header inclusion.
+#define JSON_IS_AMALGAMATION
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/version.h
+// //////////////////////////////////////////////////////////////////////
+
+#ifndef JSON_VERSION_H_INCLUDED
+#define JSON_VERSION_H_INCLUDED
+
+// Note: version must be updated in three places when doing a release. This
+// annoying process ensures that amalgamate, CMake, and meson all report the
+// correct version.
+// 1. /meson.build
+// 2. /include/json/version.h
+// 3. /CMakeLists.txt
+// IMPORTANT: also update the SOVERSION!!
+
+#define JSONCPP_VERSION_STRING "1.9.3"
+#define JSONCPP_VERSION_MAJOR 1
+#define JSONCPP_VERSION_MINOR 9
+#define JSONCPP_VERSION_PATCH 3
+#define JSONCPP_VERSION_QUALIFIER
+#define JSONCPP_VERSION_HEXA                                                   \
+  ((JSONCPP_VERSION_MAJOR << 24) | (JSONCPP_VERSION_MINOR << 16) |             \
+   (JSONCPP_VERSION_PATCH << 8))
+
+#ifdef JSONCPP_USING_SECURE_MEMORY
+#undef JSONCPP_USING_SECURE_MEMORY
+#endif
+#define JSONCPP_USING_SECURE_MEMORY 0
+// If non-zero, the library zeroes any memory that it has allocated before
+// it frees its memory.
+
+#endif // JSON_VERSION_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/version.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/allocator.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_ALLOCATOR_H_INCLUDED
+#define JSON_ALLOCATOR_H_INCLUDED
+
+#include <cstring>
+#include <memory>
+
+#pragma pack(push, 8)
+
+namespace Json {
+    template<typename T>
+    class SecureAllocator {
+    public:
+        // Type definitions
+        using value_type = T;
+        using pointer = T *;
+        using const_pointer = const T *;
+        using reference = T &;
+        using const_reference = const T &;
+        using size_type = std::size_t;
+        using difference_type = std::ptrdiff_t;
+
+        /**
+         * Allocate memory for N items using the standard allocator.
+         */
+        pointer allocate(size_type n) {
+            // allocate using "global operator new"
+            return static_cast<pointer>(::operator new(n * sizeof(T)));
+        }
+
+        /**
+         * Release memory which was allocated for N items at pointer P.
+         *
+         * The memory block is filled with zeroes before being released.
+         * The pointer argument is tagged as "volatile" to prevent the
+         * compiler optimizing out this critical step.
+         */
+        void deallocate(volatile pointer p, size_type n) {
+            std::memset(p, 0, n * sizeof(T));
+            // free using "global operator delete"
+            ::operator delete(p);
+        }
+
+        /**
+         * Construct an item in-place at pointer P.
+         */
+        template<typename... Args>
+        void construct(pointer p, Args &&... args) {
+            // construct using "placement new" and "perfect forwarding"
+            ::new(static_cast<void *>(p)) T(std::forward<Args>(args)...);
+        }
+
+        size_type max_size() const { return size_t(-1) / sizeof(T); }
+
+        pointer address(reference x) const { return std::addressof(x); }
+
+        const_pointer address(const_reference x) const { return std::addressof(x); }
+
+        /**
+         * Destroy an item in-place at pointer P.
+         */
+        void destroy(pointer p) {
+            // destroy using "explicit destructor"
+            p->~T();
+        }
+
+        // Boilerplate
+        SecureAllocator() {}
+
+        template<typename U>
+        SecureAllocator(const SecureAllocator<U> &) {}
+
+        template<typename U>
+        struct rebind {
+            using other = SecureAllocator<U>;
+        };
+    };
+
+    template<typename T, typename U>
+    bool operator==(const SecureAllocator<T> &, const SecureAllocator<U> &) {
+        return true;
+    }
+
+    template<typename T, typename U>
+    bool operator!=(const SecureAllocator<T> &, const SecureAllocator<U> &) {
+        return false;
+    }
+
+} // namespace Json
+
+#pragma pack(pop)
+
+#endif // JSON_ALLOCATOR_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/allocator.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/config.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_CONFIG_H_INCLUDED
+#define JSON_CONFIG_H_INCLUDED
+
+#include <cstddef>
+#include <cstdint>
+#include <istream>
+#include <memory>
+#include <ostream>
+#include <sstream>
+#include <string>
+#include <type_traits>
+
+// If non-zero, the library uses exceptions to report bad input instead of C
+// assertion macros. The default is to use exceptions.
+#ifndef JSON_USE_EXCEPTION
+#define JSON_USE_EXCEPTION 1
+#endif
+
+// Temporary, tracked for removal with issue #982.
+#ifndef JSON_USE_NULLREF
+#define JSON_USE_NULLREF 1
+#endif
+
+/// If defined, indicates that the source file is amalgamated
+/// to prevent private header inclusion.
+/// Remarks: it is automatically defined in the generated amalgamated header.
+// #define JSON_IS_AMALGAMATION
+
+// Export macros for DLL visibility
+#if defined(JSON_DLL_BUILD)
+#if defined(_MSC_VER) || defined(__MINGW32__)
+#define JSON_API __declspec(dllexport)
+#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
+#elif defined(__GNUC__) || defined(__clang__)
+#define JSON_API __attribute__((visibility("default")))
+#endif // if defined(_MSC_VER)
+
+#elif defined(JSON_DLL)
+#if defined(_MSC_VER) || defined(__MINGW32__)
+#define JSON_API __declspec(dllimport)
+#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
+#endif // if defined(_MSC_VER)
+#endif // ifdef JSON_DLL_BUILD
+
+#if !defined(JSON_API)
+#define JSON_API
+#endif
+
+#if defined(_MSC_VER) && _MSC_VER < 1800
+#error                                                                         \
+    "ERROR:  Visual Studio 12 (2013) with _MSC_VER=1800 is the oldest supported compiler with sufficient C++11 capabilities"
+#endif
+
+#if defined(_MSC_VER) && _MSC_VER < 1900
+// As recommended at
+// https://stackoverflow.com/questions/2915672/snprintf-and-visual-studio-2010
+extern JSON_API int msvc_pre1900_c99_snprintf(char* outBuf, size_t size,
+                                              const char* format, ...);
+#define jsoncpp_snprintf msvc_pre1900_c99_snprintf
+#else
+#define jsoncpp_snprintf std::snprintf
+#endif
+
+// If JSON_NO_INT64 is defined, then Json only support C++ "int" type for
+// integer
+// Storages, and 64 bits integer support is disabled.
+// #define JSON_NO_INT64 1
+
+// JSONCPP_OVERRIDE is maintained for backwards compatibility of external tools.
+// C++11 should be used directly in JSONCPP.
+#define JSONCPP_OVERRIDE override
+
+#ifdef __clang__
+#if __has_extension(attribute_deprecated_with_message)
+#define JSONCPP_DEPRECATED(message) __attribute__((deprecated(message)))
+#endif
+#elif defined(__GNUC__) // not clang (gcc comes later since clang emulates gcc)
+#if (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5))
+#define JSONCPP_DEPRECATED(message) __attribute__((deprecated(message)))
+#elif (__GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1))
+#define JSONCPP_DEPRECATED(message) __attribute__((__deprecated__))
+#endif                  // GNUC version
+#elif defined(_MSC_VER) // MSVC (after clang because clang on Windows emulates
+// MSVC)
+#define JSONCPP_DEPRECATED(message) __declspec(deprecated(message))
+#endif // __clang__ || __GNUC__ || _MSC_VER
+
+#if !defined(JSONCPP_DEPRECATED)
+#define JSONCPP_DEPRECATED(message)
+#endif // if !defined(JSONCPP_DEPRECATED)
+
+#if defined(__clang__) || (defined(__GNUC__) && (__GNUC__ >= 6))
+#define JSON_USE_INT64_DOUBLE_CONVERSION 1
+#endif
+
+#if !defined(JSON_IS_AMALGAMATION)
+
+#include "allocator.h"
+#include "version.h"
+
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+namespace Json {
+    using Int = int;
+    using UInt = unsigned int;
+#if defined(JSON_NO_INT64)
+    using LargestInt = int;
+    using LargestUInt = unsigned int;
+#undef JSON_HAS_INT64
+#else                 // if defined(JSON_NO_INT64)
+// For Microsoft Visual use specific types as long long is not supported
+#if defined(_MSC_VER) // Microsoft Visual Studio
+    using Int64 = __int64;
+    using UInt64 = unsigned __int64;
+#else                 // if defined(_MSC_VER) // Other platforms, use long long
+    using Int64 = int64_t;
+    using UInt64 = uint64_t;
+#endif                // if defined(_MSC_VER)
+    using LargestInt = Int64;
+    using LargestUInt = UInt64;
+#define JSON_HAS_INT64
+#endif // if defined(JSON_NO_INT64)
+
+    template<typename T>
+    using Allocator =
+    typename std::conditional<JSONCPP_USING_SECURE_MEMORY, SecureAllocator<T>,
+            std::allocator < T>>::type;
+    using String = std::basic_string<char, std::char_traits<char>, Allocator<char>>;
+    using IStringStream =
+    std::basic_istringstream<String::value_type, String::traits_type,
+            String::allocator_type>;
+    using OStringStream =
+    std::basic_ostringstream<String::value_type, String::traits_type,
+            String::allocator_type>;
+    using IStream = std::istream;
+    using OStream = std::ostream;
+} // namespace Json
+
+// Legacy names (formerly macros).
+using JSONCPP_STRING = Json::String;
+using JSONCPP_ISTRINGSTREAM = Json::IStringStream;
+using JSONCPP_OSTRINGSTREAM = Json::OStringStream;
+using JSONCPP_ISTREAM = Json::IStream;
+using JSONCPP_OSTREAM = Json::OStream;
+
+#endif // JSON_CONFIG_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/config.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/forwards.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_FORWARDS_H_INCLUDED
+#define JSON_FORWARDS_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "config.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+namespace Json {
+
+// writer.h
+    class StreamWriter;
+
+    class StreamWriterBuilder;
+
+    class Writer;
+
+    class FastWriter;
+
+    class StyledWriter;
+
+    class StyledStreamWriter;
+
+// reader.h
+    class Reader;
+
+    class CharReader;
+
+    class CharReaderBuilder;
+
+// json_features.h
+    class Features;
+
+// value.h
+    using ArrayIndex = unsigned int;
+
+    class StaticString;
+
+    class Path;
+
+    class PathArgument;
+
+    class Value;
+
+    class ValueIteratorBase;
+
+    class ValueIterator;
+
+    class ValueConstIterator;
+
+} // namespace Json
+
+#endif // JSON_FORWARDS_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/forwards.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+#endif //ifndef JSON_FORWARD_AMALGAMATED_H_INCLUDED

+ 2604 - 0
library_core/src/main/cpp/external/json/json.h

@@ -0,0 +1,2604 @@
+/// Json-cpp amalgamated header (http://jsoncpp.sourceforge.net/).
+/// It is intended to be used with #include "json/json.h"
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: LICENSE
+// //////////////////////////////////////////////////////////////////////
+
+/*
+The JsonCpp library's source code, including accompanying documentation, 
+tests and demonstration applications, are licensed under the following
+conditions...
+
+Baptiste Lepilleur and The JsonCpp Authors explicitly disclaim copyright in all 
+jurisdictions which recognize such a disclaimer. In such jurisdictions, 
+this software is released into the Public Domain.
+
+In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
+2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur and
+The JsonCpp Authors, and is released under the terms of the MIT License (see below).
+
+In jurisdictions which recognize Public Domain property, the user of this 
+software may choose to accept it either as 1) Public Domain, 2) under the 
+conditions of the MIT License (see below), or 3) under the terms of dual 
+Public Domain/MIT License conditions described here, as they choose.
+
+The MIT License is about as close to Public Domain as a license can get, and is
+described in clear, concise terms at:
+
+   http://en.wikipedia.org/wiki/MIT_License
+   
+The full text of the MIT License follows:
+
+========================================================================
+Copyright (c) 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+
+Permission is hereby granted, free of charge, to any person
+obtaining a copy of this software and associated documentation
+files (the "Software"), to deal in the Software without
+restriction, including without limitation the rights to use, copy,
+modify, merge, publish, distribute, sublicense, and/or sell copies
+of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
+BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
+ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
+========================================================================
+(END LICENSE TEXT)
+
+The MIT license is compatible with both the GPL and commercial
+software, affording one all of the rights of Public Domain with the
+minor nuisance of being required to keep the above copyright notice
+and license text in the source code. Note also that by accepting the
+Public Domain "license" you can re-license your copy using whatever
+license you like.
+
+*/
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: LICENSE
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+#ifndef JSON_AMALGAMATED_H_INCLUDED
+# define JSON_AMALGAMATED_H_INCLUDED
+/// If defined, indicates that the source file is amalgamated
+/// to prevent private header inclusion.
+#define JSON_IS_AMALGAMATION
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/version.h
+// //////////////////////////////////////////////////////////////////////
+
+#ifndef JSON_VERSION_H_INCLUDED
+#define JSON_VERSION_H_INCLUDED
+
+// Note: version must be updated in three places when doing a release. This
+// annoying process ensures that amalgamate, CMake, and meson all report the
+// correct version.
+// 1. /meson.build
+// 2. /include/json/version.h
+// 3. /CMakeLists.txt
+// IMPORTANT: also update the SOVERSION!!
+
+#define JSONCPP_VERSION_STRING "1.9.3"
+#define JSONCPP_VERSION_MAJOR 1
+#define JSONCPP_VERSION_MINOR 9
+#define JSONCPP_VERSION_PATCH 3
+#define JSONCPP_VERSION_QUALIFIER
+#define JSONCPP_VERSION_HEXA                                                   \
+  ((JSONCPP_VERSION_MAJOR << 24) | (JSONCPP_VERSION_MINOR << 16) |             \
+   (JSONCPP_VERSION_PATCH << 8))
+
+#ifdef JSONCPP_USING_SECURE_MEMORY
+#undef JSONCPP_USING_SECURE_MEMORY
+#endif
+#define JSONCPP_USING_SECURE_MEMORY 0
+// If non-zero, the library zeroes any memory that it has allocated before
+// it frees its memory.
+
+#endif // JSON_VERSION_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/version.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/allocator.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_ALLOCATOR_H_INCLUDED
+#define JSON_ALLOCATOR_H_INCLUDED
+
+#include <cstring>
+#include <memory>
+
+#pragma pack(push, 8)
+
+namespace Json {
+    template<typename T>
+    class SecureAllocator {
+    public:
+        // Type definitions
+        using value_type = T;
+        using pointer = T *;
+        using const_pointer = const T *;
+        using reference = T &;
+        using const_reference = const T &;
+        using size_type = std::size_t;
+        using difference_type = std::ptrdiff_t;
+
+        /**
+         * Allocate memory for N items using the standard allocator.
+         */
+        pointer allocate(size_type n) {
+            // allocate using "global operator new"
+            return static_cast<pointer>(::operator new(n * sizeof(T)));
+        }
+
+        /**
+         * Release memory which was allocated for N items at pointer P.
+         *
+         * The memory block is filled with zeroes before being released.
+         * The pointer argument is tagged as "volatile" to prevent the
+         * compiler optimizing out this critical step.
+         */
+        void deallocate(volatile pointer p, size_type n) {
+            std::memset(p, 0, n * sizeof(T));
+            // free using "global operator delete"
+            ::operator delete(p);
+        }
+
+        /**
+         * Construct an item in-place at pointer P.
+         */
+        template<typename... Args>
+        void construct(pointer p, Args &&... args) {
+            // construct using "placement new" and "perfect forwarding"
+            ::new(static_cast<void *>(p)) T(std::forward<Args>(args)...);
+        }
+
+        size_type max_size() const { return size_t(-1) / sizeof(T); }
+
+        pointer address(reference x) const { return std::addressof(x); }
+
+        const_pointer address(const_reference x) const { return std::addressof(x); }
+
+        /**
+         * Destroy an item in-place at pointer P.
+         */
+        void destroy(pointer p) {
+            // destroy using "explicit destructor"
+            p->~T();
+        }
+
+        // Boilerplate
+        SecureAllocator() {}
+
+        template<typename U>
+        SecureAllocator(const SecureAllocator<U> &) {}
+
+        template<typename U>
+        struct rebind {
+            using other = SecureAllocator<U>;
+        };
+    };
+
+    template<typename T, typename U>
+    bool operator==(const SecureAllocator<T> &, const SecureAllocator<U> &) {
+        return true;
+    }
+
+    template<typename T, typename U>
+    bool operator!=(const SecureAllocator<T> &, const SecureAllocator<U> &) {
+        return false;
+    }
+
+} // namespace Json
+
+#pragma pack(pop)
+
+#endif // JSON_ALLOCATOR_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/allocator.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/config.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_CONFIG_H_INCLUDED
+#define JSON_CONFIG_H_INCLUDED
+
+#include <cstddef>
+#include <cstdint>
+#include <istream>
+#include <memory>
+#include <ostream>
+#include <sstream>
+#include <string>
+#include <type_traits>
+
+// If non-zero, the library uses exceptions to report bad input instead of C
+// assertion macros. The default is to use exceptions.
+#ifndef JSON_USE_EXCEPTION
+#define JSON_USE_EXCEPTION 1
+#endif
+
+// Temporary, tracked for removal with issue #982.
+#ifndef JSON_USE_NULLREF
+#define JSON_USE_NULLREF 1
+#endif
+
+/// If defined, indicates that the source file is amalgamated
+/// to prevent private header inclusion.
+/// Remarks: it is automatically defined in the generated amalgamated header.
+// #define JSON_IS_AMALGAMATION
+
+// Export macros for DLL visibility
+#if defined(JSON_DLL_BUILD)
+#if defined(_MSC_VER) || defined(__MINGW32__)
+#define JSON_API __declspec(dllexport)
+#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
+#elif defined(__GNUC__) || defined(__clang__)
+#define JSON_API __attribute__((visibility("default")))
+#endif // if defined(_MSC_VER)
+
+#elif defined(JSON_DLL)
+#if defined(_MSC_VER) || defined(__MINGW32__)
+#define JSON_API __declspec(dllimport)
+#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
+#endif // if defined(_MSC_VER)
+#endif // ifdef JSON_DLL_BUILD
+
+#if !defined(JSON_API)
+#define JSON_API
+#endif
+
+#if defined(_MSC_VER) && _MSC_VER < 1800
+#error                                                                         \
+    "ERROR:  Visual Studio 12 (2013) with _MSC_VER=1800 is the oldest supported compiler with sufficient C++11 capabilities"
+#endif
+
+#if defined(_MSC_VER) && _MSC_VER < 1900
+// As recommended at
+// https://stackoverflow.com/questions/2915672/snprintf-and-visual-studio-2010
+extern JSON_API int msvc_pre1900_c99_snprintf(char* outBuf, size_t size,
+                                              const char* format, ...);
+#define jsoncpp_snprintf msvc_pre1900_c99_snprintf
+#else
+#define jsoncpp_snprintf std::snprintf
+#endif
+
+// If JSON_NO_INT64 is defined, then Json only support C++ "int" type for
+// integer
+// Storages, and 64 bits integer support is disabled.
+// #define JSON_NO_INT64 1
+
+// JSONCPP_OVERRIDE is maintained for backwards compatibility of external tools.
+// C++11 should be used directly in JSONCPP.
+#define JSONCPP_OVERRIDE override
+
+#ifdef __clang__
+#if __has_extension(attribute_deprecated_with_message)
+#define JSONCPP_DEPRECATED(message) __attribute__((deprecated(message)))
+#endif
+#elif defined(__GNUC__) // not clang (gcc comes later since clang emulates gcc)
+#if (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5))
+#define JSONCPP_DEPRECATED(message) __attribute__((deprecated(message)))
+#elif (__GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1))
+#define JSONCPP_DEPRECATED(message) __attribute__((__deprecated__))
+#endif                  // GNUC version
+#elif defined(_MSC_VER) // MSVC (after clang because clang on Windows emulates
+// MSVC)
+#define JSONCPP_DEPRECATED(message) __declspec(deprecated(message))
+#endif // __clang__ || __GNUC__ || _MSC_VER
+
+#if !defined(JSONCPP_DEPRECATED)
+#define JSONCPP_DEPRECATED(message)
+#endif // if !defined(JSONCPP_DEPRECATED)
+
+#if defined(__clang__) || (defined(__GNUC__) && (__GNUC__ >= 6))
+#define JSON_USE_INT64_DOUBLE_CONVERSION 1
+#endif
+
+#if !defined(JSON_IS_AMALGAMATION)
+
+#include "allocator.h"
+#include "version.h"
+
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+namespace Json {
+    using Int = int;
+    using UInt = unsigned int;
+#if defined(JSON_NO_INT64)
+    using LargestInt = int;
+    using LargestUInt = unsigned int;
+#undef JSON_HAS_INT64
+#else                 // if defined(JSON_NO_INT64)
+// For Microsoft Visual use specific types as long long is not supported
+#if defined(_MSC_VER) // Microsoft Visual Studio
+    using Int64 = __int64;
+    using UInt64 = unsigned __int64;
+#else                 // if defined(_MSC_VER) // Other platforms, use long long
+    using Int64 = int64_t;
+    using UInt64 = uint64_t;
+#endif                // if defined(_MSC_VER)
+    using LargestInt = Int64;
+    using LargestUInt = UInt64;
+#define JSON_HAS_INT64
+#endif // if defined(JSON_NO_INT64)
+
+    template<typename T>
+    using Allocator =
+    typename std::conditional<JSONCPP_USING_SECURE_MEMORY, SecureAllocator<T>,
+            std::allocator<T>>::type;
+    using String = std::basic_string<char, std::char_traits<char>, Allocator<char>>;
+    using IStringStream =
+    std::basic_istringstream<String::value_type, String::traits_type,
+            String::allocator_type>;
+    using OStringStream =
+    std::basic_ostringstream<String::value_type, String::traits_type,
+            String::allocator_type>;
+    using IStream = std::istream;
+    using OStream = std::ostream;
+} // namespace Json
+
+// Legacy names (formerly macros).
+using JSONCPP_STRING = Json::String;
+using JSONCPP_ISTRINGSTREAM = Json::IStringStream;
+using JSONCPP_OSTRINGSTREAM = Json::OStringStream;
+using JSONCPP_ISTREAM = Json::IStream;
+using JSONCPP_OSTREAM = Json::OStream;
+
+#endif // JSON_CONFIG_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/config.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/forwards.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_FORWARDS_H_INCLUDED
+#define JSON_FORWARDS_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "config.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+namespace Json {
+
+// writer.h
+    class StreamWriter;
+
+    class StreamWriterBuilder;
+
+    class Writer;
+
+    class FastWriter;
+
+    class StyledWriter;
+
+    class StyledStreamWriter;
+
+// reader.h
+    class Reader;
+
+    class CharReader;
+
+    class CharReaderBuilder;
+
+// json_features.h
+    class Features;
+
+// value.h
+    using ArrayIndex = unsigned int;
+
+    class StaticString;
+
+    class Path;
+
+    class PathArgument;
+
+    class Value;
+
+    class ValueIteratorBase;
+
+    class ValueIterator;
+
+    class ValueConstIterator;
+
+} // namespace Json
+
+#endif // JSON_FORWARDS_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/forwards.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/json_features.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_FEATURES_H_INCLUDED
+#define JSON_FEATURES_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "forwards.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+#pragma pack(push, 8)
+
+namespace Json {
+
+/** \brief Configuration passed to reader and writer.
+ * This configuration object can be used to force the Reader or Writer
+ * to behave in a standard conforming way.
+ */
+    class JSON_API Features {
+    public:
+        /** \brief A configuration that allows all features and assumes all strings
+         * are UTF-8.
+         * - C & C++ comments are allowed
+         * - Root object can be any JSON value
+         * - Assumes Value strings are encoded in UTF-8
+         */
+        static Features all();
+
+        /** \brief A configuration that is strictly compatible with the JSON
+         * specification.
+         * - Comments are forbidden.
+         * - Root object must be either an array or an object value.
+         * - Assumes Value strings are encoded in UTF-8
+         */
+        static Features strictMode();
+
+        /** \brief Initialize the configuration like JsonConfig::allFeatures;
+         */
+        Features();
+
+        /// \c true if comments are allowed. Default: \c true.
+        bool allowComments_{true};
+
+        /// \c true if root must be either an array or an object value. Default: \c
+        /// false.
+        bool strictRoot_{false};
+
+        /// \c true if dropped null placeholders are allowed. Default: \c false.
+        bool allowDroppedNullPlaceholders_{false};
+
+        /// \c true if numeric object key are allowed. Default: \c false.
+        bool allowNumericKeys_{false};
+    };
+
+} // namespace Json
+
+#pragma pack(pop)
+
+#endif // JSON_FEATURES_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/json_features.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/value.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_H_INCLUDED
+#define JSON_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "forwards.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+// Conditional NORETURN attribute on the throw functions would:
+// a) suppress false positives from static code analysis
+// b) possibly improve optimization opportunities.
+#if !defined(JSONCPP_NORETURN)
+#if defined(_MSC_VER) && _MSC_VER == 1800
+#define JSONCPP_NORETURN __declspec(noreturn)
+#else
+#define JSONCPP_NORETURN [[noreturn]]
+#endif
+#endif
+
+// Support for '= delete' with template declarations was a late addition
+// to the c++11 standard and is rejected by clang 3.8 and Apple clang 8.2
+// even though these declare themselves to be c++11 compilers.
+#if !defined(JSONCPP_TEMPLATE_DELETE)
+#if defined(__clang__) && defined(__apple_build_version__)
+#if __apple_build_version__ <= 8000042
+#define JSONCPP_TEMPLATE_DELETE
+#endif
+#elif defined(__clang__)
+#if __clang_major__ == 3 && __clang_minor__ <= 8
+#define JSONCPP_TEMPLATE_DELETE
+#endif
+#endif
+#if !defined(JSONCPP_TEMPLATE_DELETE)
+#define JSONCPP_TEMPLATE_DELETE = delete
+#endif
+#endif
+
+#include <array>
+#include <exception>
+#include <map>
+#include <memory>
+#include <string>
+#include <vector>
+
+// Disable warning C4251: <data member>: <type> needs to have dll-interface to
+// be used by...
+#if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+#pragma warning(push)
+#pragma warning(disable : 4251)
+#endif // if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+
+#pragma pack(push, 8)
+
+/** \brief JSON (JavaScript Object Notation).
+ */
+namespace Json {
+
+#if JSON_USE_EXCEPTION
+
+/** Base class for all exceptions we throw.
+ *
+ * We use nothing but these internally. Of course, STL can throw others.
+ */
+    class JSON_API Exception : public std::exception {
+    public:
+        Exception(String msg);
+
+        ~Exception() noexcept override;
+
+        char const *what() const noexcept override;
+
+    protected:
+        String msg_;
+    };
+
+/** Exceptions which the user cannot easily avoid.
+ *
+ * E.g. out-of-memory (when we use malloc), stack-overflow, malicious input
+ *
+ * \remark derived from Json::Exception
+ */
+    class JSON_API RuntimeError : public Exception {
+    public:
+        RuntimeError(String const &msg);
+    };
+
+/** Exceptions thrown by JSON_ASSERT/JSON_FAIL macros.
+ *
+ * These are precondition-violations (user bugs) and internal errors (our bugs).
+ *
+ * \remark derived from Json::Exception
+ */
+    class JSON_API LogicError : public Exception {
+    public:
+        LogicError(String const &msg);
+    };
+
+#endif
+
+/// used internally
+    JSONCPP_NORETURN void throwRuntimeError(String const &msg);
+/// used internally
+    JSONCPP_NORETURN void throwLogicError(String const &msg);
+
+/** \brief Type of the value held by a Value object.
+ */
+    enum ValueType {
+        nullValue = 0, ///< 'null' value
+        intValue,      ///< signed integer value
+        uintValue,     ///< unsigned integer value
+        realValue,     ///< double value
+        stringValue,   ///< UTF-8 string value
+        booleanValue,  ///< bool value
+        arrayValue,    ///< array value (ordered list)
+        objectValue    ///< object value (collection of name/value pairs).
+    };
+
+    enum CommentPlacement {
+        commentBefore = 0,      ///< a comment placed on the line before a value
+        commentAfterOnSameLine, ///< a comment just after a value on the same line
+        commentAfter, ///< a comment on the line after a value (only make sense for
+        /// root value)
+        numberOfCommentPlacement
+    };
+
+/** \brief Type of precision for formatting of real values.
+ */
+    enum PrecisionType {
+        significantDigits = 0, ///< we put max number of significant digits in string
+        decimalPlaces          ///< we put max number of digits after "." in string
+    };
+
+/** \brief Lightweight wrapper to tag static string.
+ *
+ * Value constructor and objectValue member assignment takes advantage of the
+ * StaticString and avoid the cost of string duplication when storing the
+ * string or the member name.
+ *
+ * Example of usage:
+ * \code
+ * Json::Value aValue( StaticString("some text") );
+ * Json::Value object;
+ * static const StaticString code("code");
+ * object[code] = 1234;
+ * \endcode
+ */
+    class JSON_API StaticString {
+    public:
+        explicit StaticString(const char *czstring) : c_str_(czstring) {}
+
+        operator const char *() const { return c_str_; }
+
+        const char *c_str() const { return c_str_; }
+
+    private:
+        const char *c_str_;
+    };
+
+/** \brief Represents a <a HREF="http://www.json.org">JSON</a> value.
+ *
+ * This class is a discriminated union wrapper that can represents a:
+ * - signed integer [range: Value::minInt - Value::maxInt]
+ * - unsigned integer (range: 0 - Value::maxUInt)
+ * - double
+ * - UTF-8 string
+ * - boolean
+ * - 'null'
+ * - an ordered list of Value
+ * - collection of name/value pairs (javascript object)
+ *
+ * The type of the held value is represented by a #ValueType and
+ * can be obtained using type().
+ *
+ * Values of an #objectValue or #arrayValue can be accessed using operator[]()
+ * methods.
+ * Non-const methods will automatically create the a #nullValue element
+ * if it does not exist.
+ * The sequence of an #arrayValue will be automatically resized and initialized
+ * with #nullValue. resize() can be used to enlarge or truncate an #arrayValue.
+ *
+ * The get() methods can be used to obtain default value in the case the
+ * required element does not exist.
+ *
+ * It is possible to iterate over the list of member keys of an object using
+ * the getMemberNames() method.
+ *
+ * \note #Value string-length fit in size_t, but keys must be < 2^30.
+ * (The reason is an implementation detail.) A #CharReader will raise an
+ * exception if a bound is exceeded to avoid security holes in your app,
+ * but the Value API does *not* check bounds. That is the responsibility
+ * of the caller.
+ */
+    class JSON_API Value {
+        friend class ValueIteratorBase;
+
+    public:
+        using Members = std::vector<String>;
+        using iterator = ValueIterator;
+        using const_iterator = ValueConstIterator;
+        using UInt = Json::UInt;
+        using Int = Json::Int;
+#if defined(JSON_HAS_INT64)
+        using UInt64 = Json::UInt64;
+        using Int64 = Json::Int64;
+#endif // defined(JSON_HAS_INT64)
+        using LargestInt = Json::LargestInt;
+        using LargestUInt = Json::LargestUInt;
+        using ArrayIndex = Json::ArrayIndex;
+
+        // Required for boost integration, e. g. BOOST_TEST
+        using value_type = std::string;
+
+#if JSON_USE_NULLREF
+        // Binary compatibility kludges, do not use.
+        static const Value &null;
+        static const Value &nullRef;
+#endif
+
+        // null and nullRef are deprecated, use this instead.
+        static Value const &nullSingleton();
+
+        /// Minimum signed integer value that can be stored in a Json::Value.
+        static constexpr LargestInt minLargestInt =
+                LargestInt(~(LargestUInt(-1) / 2));
+        /// Maximum signed integer value that can be stored in a Json::Value.
+        static constexpr LargestInt maxLargestInt = LargestInt(LargestUInt(-1) / 2);
+        /// Maximum unsigned integer value that can be stored in a Json::Value.
+        static constexpr LargestUInt maxLargestUInt = LargestUInt(-1);
+
+        /// Minimum signed int value that can be stored in a Json::Value.
+        static constexpr Int minInt = Int(~(UInt(-1) / 2));
+        /// Maximum signed int value that can be stored in a Json::Value.
+        static constexpr Int maxInt = Int(UInt(-1) / 2);
+        /// Maximum unsigned int value that can be stored in a Json::Value.
+        static constexpr UInt maxUInt = UInt(-1);
+
+#if defined(JSON_HAS_INT64)
+        /// Minimum signed 64 bits int value that can be stored in a Json::Value.
+        static constexpr Int64 minInt64 = Int64(~(UInt64(-1) / 2));
+        /// Maximum signed 64 bits int value that can be stored in a Json::Value.
+        static constexpr Int64 maxInt64 = Int64(UInt64(-1) / 2);
+        /// Maximum unsigned 64 bits int value that can be stored in a Json::Value.
+        static constexpr UInt64 maxUInt64 = UInt64(-1);
+#endif // defined(JSON_HAS_INT64)
+        /// Default precision for real value for string representation.
+        static constexpr UInt defaultRealPrecision = 17;
+        // The constant is hard-coded because some compiler have trouble
+        // converting Value::maxUInt64 to a double correctly (AIX/xlC).
+        // Assumes that UInt64 is a 64 bits integer.
+        static constexpr double maxUInt64AsDouble = 18446744073709551615.0;
+// Workaround for bug in the NVIDIAs CUDA 9.1 nvcc compiler
+// when using gcc and clang backend compilers.  CZString
+// cannot be defined as private.  See issue #486
+#ifdef __NVCC__
+        public:
+#else
+    private:
+#endif
+#ifndef JSONCPP_DOC_EXCLUDE_IMPLEMENTATION
+
+        class CZString {
+        public:
+            enum DuplicationPolicy {
+                noDuplication = 0, duplicate, duplicateOnCopy
+            };
+
+            CZString(ArrayIndex index);
+
+            CZString(char const *str, unsigned length, DuplicationPolicy allocate);
+
+            CZString(CZString const &other);
+
+            CZString(CZString &&other);
+
+            ~CZString();
+
+            CZString &operator=(const CZString &other);
+
+            CZString &operator=(CZString &&other);
+
+            bool operator<(CZString const &other) const;
+
+            bool operator==(CZString const &other) const;
+
+            ArrayIndex index() const;
+
+            // const char* c_str() const; ///< \deprecated
+            char const *data() const;
+
+            unsigned length() const;
+
+            bool isStaticString() const;
+
+        private:
+            void swap(CZString &other);
+
+            struct StringStorage {
+                unsigned policy_: 2;
+                unsigned length_: 30; // 1GB max
+            };
+
+            char const *cstr_; // actually, a prefixed string, unless policy is noDup
+            union {
+                ArrayIndex index_;
+                StringStorage storage_;
+            };
+        };
+
+    public:
+        typedef std::map<CZString, Value> ObjectValues;
+#endif // ifndef JSONCPP_DOC_EXCLUDE_IMPLEMENTATION
+
+    public:
+        /**
+         * \brief Create a default Value of the given type.
+         *
+         * This is a very useful constructor.
+         * To create an empty array, pass arrayValue.
+         * To create an empty object, pass objectValue.
+         * Another Value can then be put to this one by assignment.
+         * This is useful since clear() and resize() will not alter types.
+         *
+         * Examples:
+         *   \code
+         *   Json::Value null_value; // null
+         *   Json::Value arr_value(Json::arrayValue); // []
+         *   Json::Value obj_value(Json::objectValue); // {}
+         *   \endcode
+         */
+        Value(ValueType type = nullValue);
+
+        Value(Int value);
+
+        Value(UInt value);
+
+#if defined(JSON_HAS_INT64)
+
+        Value(Int64 value);
+
+        Value(UInt64 value);
+
+#endif // if defined(JSON_HAS_INT64)
+
+        Value(double value);
+
+        Value(const char *value); ///< Copy til first 0. (NULL causes to seg-fault.)
+        Value(const char *begin, const char *end); ///< Copy all, incl zeroes.
+        /**
+         * \brief Constructs a value from a static string.
+         *
+         * Like other value string constructor but do not duplicate the string for
+         * internal storage. The given string must remain alive after the call to
+         * this constructor.
+         *
+         * \note This works only for null-terminated strings. (We cannot change the
+         * size of this class, so we have nowhere to store the length, which might be
+         * computed later for various operations.)
+         *
+         * Example of usage:
+         *   \code
+         *   static StaticString foo("some text");
+         *   Json::Value aValue(foo);
+         *   \endcode
+         */
+        Value(const StaticString &value);
+
+        Value(const String &value);
+
+        Value(bool value);
+
+        Value(const Value &other);
+
+        Value(Value &&other);
+
+        ~Value();
+
+        /// \note Overwrite existing comments. To preserve comments, use
+        /// #swapPayload().
+        Value &operator=(const Value &other);
+
+        Value &operator=(Value &&other);
+
+        /// Swap everything.
+        void swap(Value &other);
+
+        /// Swap values but leave comments and source offsets in place.
+        void swapPayload(Value &other);
+
+        /// copy everything.
+        void copy(const Value &other);
+
+        /// copy values but leave comments and source offsets in place.
+        void copyPayload(const Value &other);
+
+        ValueType type() const;
+
+        /// Compare payload only, not comments etc.
+        bool operator<(const Value &other) const;
+
+        bool operator<=(const Value &other) const;
+
+        bool operator>=(const Value &other) const;
+
+        bool operator>(const Value &other) const;
+
+        bool operator==(const Value &other) const;
+
+        bool operator!=(const Value &other) const;
+
+        int compare(const Value &other) const;
+
+        const char *asCString() const; ///< Embedded zeroes could cause you trouble!
+#if JSONCPP_USING_SECURE_MEMORY
+        unsigned getCStringLength() const; // Allows you to understand the length of
+                                           // the CString
+#endif
+
+        String asString() const; ///< Embedded zeroes are possible.
+        /** Get raw char* of string-value.
+         *  \return false if !string. (Seg-fault if str or end are NULL.)
+         */
+        bool getString(char const **begin, char const **end) const;
+
+        Int asInt() const;
+
+        UInt asUInt() const;
+
+#if defined(JSON_HAS_INT64)
+
+        Int64 asInt64() const;
+
+        UInt64 asUInt64() const;
+
+#endif // if defined(JSON_HAS_INT64)
+
+        LargestInt asLargestInt() const;
+
+        LargestUInt asLargestUInt() const;
+
+        float asFloat() const;
+
+        double asDouble() const;
+
+        bool asBool() const;
+
+        bool isNull() const;
+
+        bool isBool() const;
+
+        bool isInt() const;
+
+        bool isInt64() const;
+
+        bool isUInt() const;
+
+        bool isUInt64() const;
+
+        bool isIntegral() const;
+
+        bool isDouble() const;
+
+        bool isNumeric() const;
+
+        bool isString() const;
+
+        bool isArray() const;
+
+        bool isObject() const;
+
+        /// The `as<T>` and `is<T>` member function templates and specializations.
+        template<typename T>
+        T as() const JSONCPP_TEMPLATE_DELETE;
+
+        template<typename T>
+        bool is() const JSONCPP_TEMPLATE_DELETE;
+
+        bool isConvertibleTo(ValueType other) const;
+
+        /// Number of values in array or object
+        ArrayIndex size() const;
+
+        /// \brief Return true if empty array, empty object, or null;
+        /// otherwise, false.
+        bool empty() const;
+
+        /// Return !isNull()
+        explicit operator bool() const;
+
+        /// Remove all object members and array elements.
+        /// \pre type() is arrayValue, objectValue, or nullValue
+        /// \post type() is unchanged
+        void clear();
+
+        /// Resize the array to newSize elements.
+        /// New elements are initialized to null.
+        /// May only be called on nullValue or arrayValue.
+        /// \pre type() is arrayValue or nullValue
+        /// \post type() is arrayValue
+        void resize(ArrayIndex newSize);
+
+        //@{
+        /// Access an array element (zero based index). If the array contains less
+        /// than index element, then null value are inserted in the array so that
+        /// its size is index+1.
+        /// (You may need to say 'value[0u]' to get your compiler to distinguish
+        /// this from the operator[] which takes a string.)
+        Value &operator[](ArrayIndex index);
+
+        Value &operator[](int index);
+        //@}
+
+        //@{
+        /// Access an array element (zero based index).
+        /// (You may need to say 'value[0u]' to get your compiler to distinguish
+        /// this from the operator[] which takes a string.)
+        const Value &operator[](ArrayIndex index) const;
+
+        const Value &operator[](int index) const;
+        //@}
+
+        /// If the array contains at least index+1 elements, returns the element
+        /// value, otherwise returns defaultValue.
+        Value get(ArrayIndex index, const Value &defaultValue) const;
+
+        /// Return true if index < size().
+        bool isValidIndex(ArrayIndex index) const;
+
+        /// \brief Append value to array at the end.
+        ///
+        /// Equivalent to jsonvalue[jsonvalue.size()] = value;
+        Value &append(const Value &value);
+
+        Value &append(Value &&value);
+
+        /// \brief Insert value in array at specific index
+        bool insert(ArrayIndex index, const Value &newValue);
+
+        bool insert(ArrayIndex index, Value &&newValue);
+
+        /// Access an object value by name, create a null member if it does not exist.
+        /// \note Because of our implementation, keys are limited to 2^30 -1 chars.
+        /// Exceeding that will cause an exception.
+        Value &operator[](const char *key);
+
+        /// Access an object value by name, returns null if there is no member with
+        /// that name.
+        const Value &operator[](const char *key) const;
+
+        /// Access an object value by name, create a null member if it does not exist.
+        /// \param key may contain embedded nulls.
+        Value &operator[](const String &key);
+
+        /// Access an object value by name, returns null if there is no member with
+        /// that name.
+        /// \param key may contain embedded nulls.
+        const Value &operator[](const String &key) const;
+
+        /** \brief Access an object value by name, create a null member if it does not
+         * exist.
+         *
+         * If the object has no entry for that name, then the member name used to
+         * store the new entry is not duplicated.
+         * Example of use:
+         *   \code
+         *   Json::Value object;
+         *   static const StaticString code("code");
+         *   object[code] = 1234;
+         *   \endcode
+         */
+        Value &operator[](const StaticString &key);
+
+        /// Return the member named key if it exist, defaultValue otherwise.
+        /// \note deep copy
+        Value get(const char *key, const Value &defaultValue) const;
+
+        /// Return the member named key if it exist, defaultValue otherwise.
+        /// \note deep copy
+        /// \note key may contain embedded nulls.
+        Value get(const char *begin, const char *end,
+                  const Value &defaultValue) const;
+
+        /// Return the member named key if it exist, defaultValue otherwise.
+        /// \note deep copy
+        /// \param key may contain embedded nulls.
+        Value get(const String &key, const Value &defaultValue) const;
+
+        /// Most general and efficient version of isMember()const, get()const,
+        /// and operator[]const
+        /// \note As stated elsewhere, behavior is undefined if (end-begin) >= 2^30
+        Value const *find(char const *begin, char const *end) const;
+
+        /// Most general and efficient version of object-mutators.
+        /// \note As stated elsewhere, behavior is undefined if (end-begin) >= 2^30
+        /// \return non-zero, but JSON_ASSERT if this is neither object nor nullValue.
+        Value *demand(char const *begin, char const *end);
+
+        /// \brief Remove and return the named member.
+        ///
+        /// Do nothing if it did not exist.
+        /// \pre type() is objectValue or nullValue
+        /// \post type() is unchanged
+        void removeMember(const char *key);
+
+        /// Same as removeMember(const char*)
+        /// \param key may contain embedded nulls.
+        void removeMember(const String &key);
+
+        /// Same as removeMember(const char* begin, const char* end, Value* removed),
+        /// but 'key' is null-terminated.
+        bool removeMember(const char *key, Value *removed);
+
+        /** \brief Remove the named map member.
+         *
+         *  Update 'removed' iff removed.
+         *  \param key may contain embedded nulls.
+         *  \return true iff removed (no exceptions)
+         */
+        bool removeMember(String const &key, Value *removed);
+
+        /// Same as removeMember(String const& key, Value* removed)
+        bool removeMember(const char *begin, const char *end, Value *removed);
+
+        /** \brief Remove the indexed array element.
+         *
+         *  O(n) expensive operations.
+         *  Update 'removed' iff removed.
+         *  \return true if removed (no exceptions)
+         */
+        bool removeIndex(ArrayIndex index, Value *removed);
+
+        /// Return true if the object has a member named key.
+        /// \note 'key' must be null-terminated.
+        bool isMember(const char *key) const;
+
+        /// Return true if the object has a member named key.
+        /// \param key may contain embedded nulls.
+        bool isMember(const String &key) const;
+
+        /// Same as isMember(String const& key)const
+        bool isMember(const char *begin, const char *end) const;
+
+        /// \brief Return a list of the member names.
+        ///
+        /// If null, return an empty list.
+        /// \pre type() is objectValue or nullValue
+        /// \post if type() was nullValue, it remains nullValue
+        Members getMemberNames() const;
+
+        /// \deprecated Always pass len.
+        JSONCPP_DEPRECATED("Use setComment(String const&) instead.")
+
+        void setComment(const char *comment, CommentPlacement placement) {
+            setComment(String(comment, strlen(comment)), placement);
+        }
+
+        /// Comments must be //... or /* ... */
+        void setComment(const char *comment, size_t len, CommentPlacement placement) {
+            setComment(String(comment, len), placement);
+        }
+
+        /// Comments must be //... or /* ... */
+        void setComment(String comment, CommentPlacement placement);
+
+        bool hasComment(CommentPlacement placement) const;
+
+        /// Include delimiters and embedded newlines.
+        String getComment(CommentPlacement placement) const;
+
+        String toStyledString() const;
+
+        const_iterator begin() const;
+
+        const_iterator end() const;
+
+        iterator begin();
+
+        iterator end();
+
+        // Accessors for the [start, limit) range of bytes within the JSON text from
+        // which this value was parsed, if any.
+        void setOffsetStart(ptrdiff_t start);
+
+        void setOffsetLimit(ptrdiff_t limit);
+
+        ptrdiff_t getOffsetStart() const;
+
+        ptrdiff_t getOffsetLimit() const;
+
+    private:
+        void setType(ValueType v) {
+            bits_.value_type_ = static_cast<unsigned char>(v);
+        }
+
+        bool isAllocated() const { return bits_.allocated_; }
+
+        void setIsAllocated(bool v) { bits_.allocated_ = v; }
+
+        void initBasic(ValueType type, bool allocated = false);
+
+        void dupPayload(const Value &other);
+
+        void releasePayload();
+
+        void dupMeta(const Value &other);
+
+        Value &resolveReference(const char *key);
+
+        Value &resolveReference(const char *key, const char *end);
+
+        // struct MemberNamesTransform
+        //{
+        //   typedef const char *result_type;
+        //   const char *operator()( const CZString &name ) const
+        //   {
+        //      return name.c_str();
+        //   }
+        //};
+
+        union ValueHolder {
+            LargestInt int_;
+            LargestUInt uint_;
+            double real_;
+            bool bool_;
+            char *string_; // if allocated_, ptr to { unsigned, char[] }.
+            ObjectValues *map_;
+        } value_;
+
+        struct {
+            // Really a ValueType, but types should agree for bitfield packing.
+            unsigned int value_type_: 8;
+            // Unless allocated_, string_ must be null-terminated.
+            unsigned int allocated_: 1;
+        } bits_;
+
+        class Comments {
+        public:
+            Comments() = default;
+
+            Comments(const Comments &that);
+
+            Comments(Comments &&that);
+
+            Comments &operator=(const Comments &that);
+
+            Comments &operator=(Comments &&that);
+
+            bool has(CommentPlacement slot) const;
+
+            String get(CommentPlacement slot) const;
+
+            void set(CommentPlacement slot, String comment);
+
+        private:
+            using Array = std::array<String, numberOfCommentPlacement>;
+            std::unique_ptr<Array> ptr_;
+        };
+
+        Comments comments_;
+
+        // [start, limit) byte offsets in the source JSON text from which this Value
+        // was extracted.
+        ptrdiff_t start_;
+        ptrdiff_t limit_;
+    };
+
+    template<>
+    inline bool Value::as<bool>() const { return asBool(); }
+
+    template<>
+    inline bool Value::is<bool>() const { return isBool(); }
+
+    template<>
+    inline Int Value::as<Int>() const { return asInt(); }
+
+    template<>
+    inline bool Value::is<Int>() const { return isInt(); }
+
+    template<>
+    inline UInt Value::as<UInt>() const { return asUInt(); }
+
+    template<>
+    inline bool Value::is<UInt>() const { return isUInt(); }
+
+#if defined(JSON_HAS_INT64)
+
+    template<>
+    inline Int64 Value::as<Int64>() const { return asInt64(); }
+
+    template<>
+    inline bool Value::is<Int64>() const { return isInt64(); }
+
+    template<>
+    inline UInt64 Value::as<UInt64>() const { return asUInt64(); }
+
+    template<>
+    inline bool Value::is<UInt64>() const { return isUInt64(); }
+
+#endif
+
+    template<>
+    inline double Value::as<double>() const { return asDouble(); }
+
+    template<>
+    inline bool Value::is<double>() const { return isDouble(); }
+
+    template<>
+    inline String Value::as<String>() const { return asString(); }
+
+    template<>
+    inline bool Value::is<String>() const { return isString(); }
+
+/// These `as` specializations are type conversions, and do not have a
+/// corresponding `is`.
+    template<>
+    inline float Value::as<float>() const { return asFloat(); }
+
+    template<>
+    inline const char *Value::as<const char *>() const {
+        return asCString();
+    }
+
+/** \brief Experimental and untested: represents an element of the "path" to
+ * access a node.
+ */
+    class JSON_API PathArgument {
+    public:
+        friend class Path;
+
+        PathArgument();
+
+        PathArgument(ArrayIndex index);
+
+        PathArgument(const char *key);
+
+        PathArgument(String key);
+
+    private:
+        enum Kind {
+            kindNone = 0, kindIndex, kindKey
+        };
+        String key_;
+        ArrayIndex index_{};
+        Kind kind_{kindNone};
+    };
+
+/** \brief Experimental and untested: represents a "path" to access a node.
+ *
+ * Syntax:
+ * - "." => root node
+ * - ".[n]" => elements at index 'n' of root node (an array value)
+ * - ".name" => member named 'name' of root node (an object value)
+ * - ".name1.name2.name3"
+ * - ".[0][1][2].name1[3]"
+ * - ".%" => member name is provided as parameter
+ * - ".[%]" => index is provided as parameter
+ */
+    class JSON_API Path {
+    public:
+        Path(const String &path, const PathArgument &a1 = PathArgument(),
+             const PathArgument &a2 = PathArgument(),
+             const PathArgument &a3 = PathArgument(),
+             const PathArgument &a4 = PathArgument(),
+             const PathArgument &a5 = PathArgument());
+
+        const Value &resolve(const Value &root) const;
+
+        Value resolve(const Value &root, const Value &defaultValue) const;
+
+        /// Creates the "path" to access the specified node and returns a reference on
+        /// the node.
+        Value &make(Value &root) const;
+
+    private:
+        using InArgs = std::vector<const PathArgument *>;
+        using Args = std::vector<PathArgument>;
+
+        void makePath(const String &path, const InArgs &in);
+
+        void addPathInArg(const String &path, const InArgs &in,
+                          InArgs::const_iterator &itInArg, PathArgument::Kind kind);
+
+        static void invalidPath(const String &path, int location);
+
+        Args args_;
+    };
+
+/** \brief base class for Value iterators.
+ *
+ */
+    class JSON_API ValueIteratorBase {
+    public:
+        using iterator_category = std::bidirectional_iterator_tag;
+        using size_t = unsigned int;
+        using difference_type = int;
+        using SelfType = ValueIteratorBase;
+
+        bool operator==(const SelfType &other) const { return isEqual(other); }
+
+        bool operator!=(const SelfType &other) const { return !isEqual(other); }
+
+        difference_type operator-(const SelfType &other) const {
+            return other.computeDistance(*this);
+        }
+
+        /// Return either the index or the member name of the referenced value as a
+        /// Value.
+        Value key() const;
+
+        /// Return the index of the referenced Value, or -1 if it is not an
+        /// arrayValue.
+        UInt index() const;
+
+        /// Return the member name of the referenced Value, or "" if it is not an
+        /// objectValue.
+        /// \note Avoid `c_str()` on result, as embedded zeroes are possible.
+        String name() const;
+
+        /// Return the member name of the referenced Value. "" if it is not an
+        /// objectValue.
+        /// \deprecated This cannot be used for UTF-8 strings, since there can be
+        /// embedded nulls.
+        JSONCPP_DEPRECATED("Use `key = name();` instead.")
+
+        char const *memberName() const;
+
+        /// Return the member name of the referenced Value, or NULL if it is not an
+        /// objectValue.
+        /// \note Better version than memberName(). Allows embedded nulls.
+        char const *memberName(char const **end) const;
+
+    protected:
+        /*! Internal utility functions to assist with implementing
+         *   other iterator functions. The const and non-const versions
+         *   of the "deref" protected methods expose the protected
+         *   current_ member variable in a way that can often be
+         *   optimized away by the compiler.
+         */
+        const Value &deref() const;
+
+        Value &deref();
+
+        void increment();
+
+        void decrement();
+
+        difference_type computeDistance(const SelfType &other) const;
+
+        bool isEqual(const SelfType &other) const;
+
+        void copy(const SelfType &other);
+
+    private:
+        Value::ObjectValues::iterator current_;
+        // Indicates that iterator is for a null value.
+        bool isNull_{true};
+
+    public:
+        // For some reason, BORLAND needs these at the end, rather
+        // than earlier. No idea why.
+        ValueIteratorBase();
+
+        explicit ValueIteratorBase(const Value::ObjectValues::iterator &current);
+    };
+
+/** \brief const iterator for object and array value.
+ *
+ */
+    class JSON_API ValueConstIterator : public ValueIteratorBase {
+        friend class Value;
+
+    public:
+        using value_type = const Value;
+        // typedef unsigned int size_t;
+        // typedef int difference_type;
+        using reference = const Value &;
+        using pointer = const Value *;
+        using SelfType = ValueConstIterator;
+
+        ValueConstIterator();
+
+        ValueConstIterator(ValueIterator const &other);
+
+    private:
+        /*! \internal Use by Value to create an iterator.
+         */
+        explicit ValueConstIterator(const Value::ObjectValues::iterator &current);
+
+    public:
+        SelfType &operator=(const ValueIteratorBase &other);
+
+        SelfType operator++(int) {
+            SelfType temp(*this);
+            ++*this;
+            return temp;
+        }
+
+        SelfType operator--(int) {
+            SelfType temp(*this);
+            --*this;
+            return temp;
+        }
+
+        SelfType &operator--() {
+            decrement();
+            return *this;
+        }
+
+        SelfType &operator++() {
+            increment();
+            return *this;
+        }
+
+        reference operator*() const { return deref(); }
+
+        pointer operator->() const { return &deref(); }
+    };
+
+/** \brief Iterator for object and array value.
+ */
+    class JSON_API ValueIterator : public ValueIteratorBase {
+        friend class Value;
+
+    public:
+        using value_type = Value;
+        using size_t = unsigned int;
+        using difference_type = int;
+        using reference = Value &;
+        using pointer = Value *;
+        using SelfType = ValueIterator;
+
+        ValueIterator();
+
+        explicit ValueIterator(const ValueConstIterator &other);
+
+        ValueIterator(const ValueIterator &other);
+
+    private:
+        /*! \internal Use by Value to create an iterator.
+         */
+        explicit ValueIterator(const Value::ObjectValues::iterator &current);
+
+    public:
+        SelfType &operator=(const SelfType &other);
+
+        SelfType operator++(int) {
+            SelfType temp(*this);
+            ++*this;
+            return temp;
+        }
+
+        SelfType operator--(int) {
+            SelfType temp(*this);
+            --*this;
+            return temp;
+        }
+
+        SelfType &operator--() {
+            decrement();
+            return *this;
+        }
+
+        SelfType &operator++() {
+            increment();
+            return *this;
+        }
+
+        /*! The return value of non-const iterators can be
+         *  changed, so the these functions are not const
+         *  because the returned references/pointers can be used
+         *  to change state of the base class.
+         */
+        reference operator*() { return deref(); }
+
+        pointer operator->() { return &deref(); }
+    };
+
+    inline void swap(Value &a, Value &b) { a.swap(b); }
+
+} // namespace Json
+
+#pragma pack(pop)
+
+#if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+#pragma warning(pop)
+#endif // if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+
+#endif // JSON_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/value.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/reader.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_READER_H_INCLUDED
+#define JSON_READER_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "json_features.h"
+#include "value.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+#include <deque>
+#include <iosfwd>
+#include <istream>
+#include <stack>
+#include <string>
+
+// Disable warning C4251: <data member>: <type> needs to have dll-interface to
+// be used by...
+#if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+#pragma warning(push)
+#pragma warning(disable : 4251)
+#endif // if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+
+#pragma pack(push, 8)
+
+namespace Json {
+
+/** \brief Unserialize a <a HREF="http://www.json.org">JSON</a> document into a
+ * Value.
+ *
+ * \deprecated Use CharReader and CharReaderBuilder.
+ */
+
+    class JSONCPP_DEPRECATED(
+            "Use CharReader and CharReaderBuilder instead.") JSON_API Reader {
+    public:
+        using Char = char;
+        using Location = const Char *;
+
+        /** \brief An error tagged with where in the JSON text it was encountered.
+         *
+         * The offsets give the [start, limit) range of bytes within the text. Note
+         * that this is bytes, not codepoints.
+         */
+        struct StructuredError {
+            ptrdiff_t offset_start;
+            ptrdiff_t offset_limit;
+            String message;
+        };
+
+        /** \brief Constructs a Reader allowing all features for parsing.
+         */
+        JSONCPP_DEPRECATED("Use CharReader and CharReaderBuilder instead")
+
+        Reader();
+
+        /** \brief Constructs a Reader allowing the specified feature put for parsing.
+         */
+        JSONCPP_DEPRECATED("Use CharReader and CharReaderBuilder instead")
+
+        Reader(const Features &features);
+
+        /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a>
+         * document.
+         *
+         * \param      document        UTF-8 encoded string containing the document
+         *                             to read.
+         * \param[out] root            Contains the root value of the document if it
+         *                             was successfully parsed.
+         * \param      collectComments \c true to collect comment and allow writing
+         *                             them back during serialization, \c false to
+         *                             discard comments.  This parameter is ignored
+         *                             if Features::allowComments_ is \c false.
+         * \return \c true if the document was successfully parsed, \c false if an
+         * error occurred.
+         */
+        bool parse(const std::string &document, Value &root,
+                   bool collectComments = true);
+
+        /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a>
+         * document.
+         *
+         * \param      beginDoc        Pointer on the beginning of the UTF-8 encoded
+         *                             string of the document to read.
+         * \param      endDoc          Pointer on the end of the UTF-8 encoded string
+         *                             of the document to read.  Must be >= beginDoc.
+         * \param[out] root            Contains the root value of the document if it
+         *                             was successfully parsed.
+         * \param      collectComments \c true to collect comment and allow writing
+         *                             them back during serialization, \c false to
+         *                             discard comments.  This parameter is ignored
+         *                             if Features::allowComments_ is \c false.
+         * \return \c true if the document was successfully parsed, \c false if an
+         * error occurred.
+         */
+        bool parse(const char *beginDoc, const char *endDoc, Value &root,
+                   bool collectComments = true);
+
+        /// \brief Parse from input stream.
+        /// \see Json::operator>>(std::istream&, Json::Value&).
+        bool parse(IStream &is, Value &root, bool collectComments = true);
+
+        /** \brief Returns a user friendly string that list errors in the parsed
+         * document.
+         *
+         * \return Formatted error message with the list of errors with their
+         * location in the parsed document. An empty string is returned if no error
+         * occurred during parsing.
+         * \deprecated Use getFormattedErrorMessages() instead (typo fix).
+         */
+        JSONCPP_DEPRECATED("Use getFormattedErrorMessages() instead.")
+
+        String getFormatedErrorMessages() const;
+
+        /** \brief Returns a user friendly string that list errors in the parsed
+         * document.
+         *
+         * \return Formatted error message with the list of errors with their
+         * location in the parsed document. An empty string is returned if no error
+         * occurred during parsing.
+         */
+        String getFormattedErrorMessages() const;
+
+        /** \brief Returns a vector of structured errors encountered while parsing.
+         *
+         * \return A (possibly empty) vector of StructuredError objects. Currently
+         * only one error can be returned, but the caller should tolerate multiple
+         * errors.  This can occur if the parser recovers from a non-fatal parse
+         * error and then encounters additional errors.
+         */
+        std::vector<StructuredError> getStructuredErrors() const;
+
+        /** \brief Add a semantic error message.
+         *
+         * \param value   JSON Value location associated with the error
+         * \param message The error message.
+         * \return \c true if the error was successfully added, \c false if the Value
+         * offset exceeds the document size.
+         */
+        bool pushError(const Value &value, const String &message);
+
+        /** \brief Add a semantic error message with extra context.
+         *
+         * \param value   JSON Value location associated with the error
+         * \param message The error message.
+         * \param extra   Additional JSON Value location to contextualize the error
+         * \return \c true if the error was successfully added, \c false if either
+         * Value offset exceeds the document size.
+         */
+        bool pushError(const Value &value, const String &message, const Value &extra);
+
+        /** \brief Return whether there are any errors.
+         *
+         * \return \c true if there are no errors to report \c false if errors have
+         * occurred.
+         */
+        bool good() const;
+
+    private:
+        enum TokenType {
+            tokenEndOfStream = 0,
+            tokenObjectBegin,
+            tokenObjectEnd,
+            tokenArrayBegin,
+            tokenArrayEnd,
+            tokenString,
+            tokenNumber,
+            tokenTrue,
+            tokenFalse,
+            tokenNull,
+            tokenArraySeparator,
+            tokenMemberSeparator,
+            tokenComment,
+            tokenError
+        };
+
+        class Token {
+        public:
+            TokenType type_;
+            Location start_;
+            Location end_;
+        };
+
+        class ErrorInfo {
+        public:
+            Token token_;
+            String message_;
+            Location extra_;
+        };
+
+        using Errors = std::deque<ErrorInfo>;
+
+        bool readToken(Token &token);
+
+        void skipSpaces();
+
+        bool match(const Char *pattern, int patternLength);
+
+        bool readComment();
+
+        bool readCStyleComment();
+
+        bool readCppStyleComment();
+
+        bool readString();
+
+        void readNumber();
+
+        bool readValue();
+
+        bool readObject(Token &token);
+
+        bool readArray(Token &token);
+
+        bool decodeNumber(Token &token);
+
+        bool decodeNumber(Token &token, Value &decoded);
+
+        bool decodeString(Token &token);
+
+        bool decodeString(Token &token, String &decoded);
+
+        bool decodeDouble(Token &token);
+
+        bool decodeDouble(Token &token, Value &decoded);
+
+        bool decodeUnicodeCodePoint(Token &token, Location &current, Location end,
+                                    unsigned int &unicode);
+
+        bool decodeUnicodeEscapeSequence(Token &token, Location &current,
+                                         Location end, unsigned int &unicode);
+
+        bool addError(const String &message, Token &token, Location extra = nullptr);
+
+        bool recoverFromError(TokenType skipUntilToken);
+
+        bool addErrorAndRecover(const String &message, Token &token,
+                                TokenType skipUntilToken);
+
+        void skipUntilSpace();
+
+        Value &currentValue();
+
+        Char getNextChar();
+
+        void getLocationLineAndColumn(Location location, int &line,
+                                      int &column) const;
+
+        String getLocationLineAndColumn(Location location) const;
+
+        void addComment(Location begin, Location end, CommentPlacement placement);
+
+        void skipCommentTokens(Token &token);
+
+        static bool containsNewLine(Location begin, Location end);
+
+        static String normalizeEOL(Location begin, Location end);
+
+        using Nodes = std::stack<Value *>;
+        Nodes nodes_;
+        Errors errors_;
+        String document_;
+        Location begin_{};
+        Location end_{};
+        Location current_{};
+        Location lastValueEnd_{};
+        Value *lastValue_{};
+        String commentsBefore_;
+        Features features_;
+        bool collectComments_{};
+    }; // Reader
+
+/** Interface for reading JSON from a char array.
+ */
+    class JSON_API CharReader {
+    public:
+        virtual ~CharReader() = default;
+
+        /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a>
+         * document. The document must be a UTF-8 encoded string containing the
+         * document to read.
+         *
+         * \param      beginDoc Pointer on the beginning of the UTF-8 encoded string
+         *                      of the document to read.
+         * \param      endDoc   Pointer on the end of the UTF-8 encoded string of the
+         *                      document to read. Must be >= beginDoc.
+         * \param[out] root     Contains the root value of the document if it was
+         *                      successfully parsed.
+         * \param[out] errs     Formatted error messages (if not NULL) a user
+         *                      friendly string that lists errors in the parsed
+         *                      document.
+         * \return \c true if the document was successfully parsed, \c false if an
+         * error occurred.
+         */
+        virtual bool parse(char const *beginDoc, char const *endDoc, Value *root,
+                           String *errs) = 0;
+
+        class JSON_API Factory {
+        public:
+            virtual ~Factory() = default;
+
+            /** \brief Allocate a CharReader via operator new().
+             * \throw std::exception if something goes wrong (e.g. invalid settings)
+             */
+            virtual CharReader *newCharReader() const = 0;
+        }; // Factory
+    };   // CharReader
+
+/** \brief Build a CharReader implementation.
+ *
+ * Usage:
+ *   \code
+ *   using namespace Json;
+ *   CharReaderBuilder builder;
+ *   builder["collectComments"] = false;
+ *   Value value;
+ *   String errs;
+ *   bool ok = parseFromStream(builder, std::cin, &value, &errs);
+ *   \endcode
+ */
+    class JSON_API CharReaderBuilder : public CharReader::Factory {
+    public:
+        // Note: We use a Json::Value so that we can add data-members to this class
+        // without a major version bump.
+        /** Configuration of this builder.
+         * These are case-sensitive.
+         * Available settings (case-sensitive):
+         * - `"collectComments": false or true`
+         *   - true to collect comment and allow writing them back during
+         *     serialization, false to discard comments.  This parameter is ignored
+         *     if allowComments is false.
+         * - `"allowComments": false or true`
+         *   - true if comments are allowed.
+         * - `"allowTrailingCommas": false or true`
+         *   - true if trailing commas in objects and arrays are allowed.
+         * - `"strictRoot": false or true`
+         *   - true if root must be either an array or an object value
+         * - `"allowDroppedNullPlaceholders": false or true`
+         *   - true if dropped null placeholders are allowed. (See
+         *     StreamWriterBuilder.)
+         * - `"allowNumericKeys": false or true`
+         *   - true if numeric object keys are allowed.
+         * - `"allowSingleQuotes": false or true`
+         *   - true if '' are allowed for strings (both keys and values)
+         * - `"stackLimit": integer`
+         *   - Exceeding stackLimit (recursive depth of `readValue()`) will cause an
+         *     exception.
+         *   - This is a security issue (seg-faults caused by deeply nested JSON), so
+         *     the default is low.
+         * - `"failIfExtra": false or true`
+         *   - If true, `parse()` returns false when extra non-whitespace trails the
+         *     JSON value in the input string.
+         * - `"rejectDupKeys": false or true`
+         *   - If true, `parse()` returns false when a key is duplicated within an
+         *     object.
+         * - `"allowSpecialFloats": false or true`
+         *   - If true, special float values (NaNs and infinities) are allowed and
+         *     their values are lossfree restorable.
+         *
+         * You can examine 'settings_` yourself to see the defaults. You can also
+         * write and read them just like any JSON Value.
+         * \sa setDefaults()
+         */
+        Json::Value settings_;
+
+        CharReaderBuilder();
+
+        ~CharReaderBuilder() override;
+
+        CharReader *newCharReader() const override;
+
+        /** \return true if 'settings' are legal and consistent;
+         *   otherwise, indicate bad settings via 'invalid'.
+         */
+        bool validate(Json::Value *invalid) const;
+
+        /** A simple way to update a specific setting.
+         */
+        Value &operator[](const String &key);
+
+        /** Called by ctor, but you can use this to reset settings_.
+         * \pre 'settings' != NULL (but Json::null is fine)
+         * \remark Defaults:
+         * \snippet src/lib_json/json_reader.cpp CharReaderBuilderDefaults
+         */
+        static void setDefaults(Json::Value *settings);
+
+        /** Same as old Features::strictMode().
+         * \pre 'settings' != NULL (but Json::null is fine)
+         * \remark Defaults:
+         * \snippet src/lib_json/json_reader.cpp CharReaderBuilderStrictMode
+         */
+        static void strictMode(Json::Value *settings);
+    };
+
+/** Consume entire stream and use its begin/end.
+ * Someday we might have a real StreamReader, but for now this
+ * is convenient.
+ */
+    bool JSON_API parseFromStream(CharReader::Factory const &, IStream &, Value *root,
+                                  String *errs);
+
+/** \brief Read from 'sin' into 'root'.
+ *
+ * Always keep comments from the input JSON.
+ *
+ * This can be used to read a file into a particular sub-object.
+ * For example:
+ *   \code
+ *   Json::Value root;
+ *   cin >> root["dir"]["file"];
+ *   cout << root;
+ *   \endcode
+ * Result:
+ * \verbatim
+ * {
+ * "dir": {
+ *    "file": {
+ *    // The input stream JSON would be nested here.
+ *    }
+ * }
+ * }
+ * \endverbatim
+ * \throw std::exception on parse error.
+ * \see Json::operator<<()
+ */
+    JSON_API IStream &operator>>(IStream &, Value &);
+
+} // namespace Json
+
+#pragma pack(pop)
+
+#if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+#pragma warning(pop)
+#endif // if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+
+#endif // JSON_READER_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/reader.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/writer.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_WRITER_H_INCLUDED
+#define JSON_WRITER_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "value.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+#include <ostream>
+#include <string>
+#include <vector>
+
+// Disable warning C4251: <data member>: <type> needs to have dll-interface to
+// be used by...
+#if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING) && defined(_MSC_VER)
+#pragma warning(push)
+#pragma warning(disable : 4251)
+#endif // if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+
+#pragma pack(push, 8)
+
+namespace Json {
+
+    class Value;
+
+/**
+ *
+ * Usage:
+ *  \code
+ *  using namespace Json;
+ *  void writeToStdout(StreamWriter::Factory const& factory, Value const& value)
+ * { std::unique_ptr<StreamWriter> const writer( factory.newStreamWriter());
+ *    writer->write(value, &std::cout);
+ *    std::cout << std::endl;  // add lf and flush
+ *  }
+ *  \endcode
+ */
+    class JSON_API StreamWriter {
+    protected:
+        OStream *sout_; // not owned; will not delete
+    public:
+        StreamWriter();
+
+        virtual ~StreamWriter();
+
+        /** Write Value into document as configured in sub-class.
+         *   Do not take ownership of sout, but maintain a reference during function.
+         *   \pre sout != NULL
+         *   \return zero on success (For now, we always return zero, so check the
+         *   stream instead.) \throw std::exception possibly, depending on
+         * configuration
+         */
+        virtual int write(Value const &root, OStream *sout) = 0;
+
+        /** \brief A simple abstract factory.
+         */
+        class JSON_API Factory {
+        public:
+            virtual ~Factory();
+
+            /** \brief Allocate a CharReader via operator new().
+             * \throw std::exception if something goes wrong (e.g. invalid settings)
+             */
+            virtual StreamWriter *newStreamWriter() const = 0;
+        }; // Factory
+    };   // StreamWriter
+
+/** \brief Write into stringstream, then return string, for convenience.
+ * A StreamWriter will be created from the factory, used, and then deleted.
+ */
+    String JSON_API writeString(StreamWriter::Factory const &factory,
+                                Value const &root);
+
+/** \brief Build a StreamWriter implementation.
+
+* Usage:
+*   \code
+*   using namespace Json;
+*   Value value = ...;
+*   StreamWriterBuilder builder;
+*   builder["commentStyle"] = "None";
+*   builder["indentation"] = "   ";  // or whatever you like
+*   std::unique_ptr<Json::StreamWriter> writer(
+*      builder.newStreamWriter());
+*   writer->write(value, &std::cout);
+*   std::cout << std::endl;  // add lf and flush
+*   \endcode
+*/
+    class JSON_API StreamWriterBuilder : public StreamWriter::Factory {
+    public:
+        // Note: We use a Json::Value so that we can add data-members to this class
+        // without a major version bump.
+        /** Configuration of this builder.
+         *  Available settings (case-sensitive):
+         *  - "commentStyle": "None" or "All"
+         *  - "indentation":  "<anything>".
+         *  - Setting this to an empty string also omits newline characters.
+         *  - "enableYAMLCompatibility": false or true
+         *  - slightly change the whitespace around colons
+         *  - "dropNullPlaceholders": false or true
+         *  - Drop the "null" string from the writer's output for nullValues.
+         *    Strictly speaking, this is not valid JSON. But when the output is being
+         *    fed to a browser's JavaScript, it makes for smaller output and the
+         *    browser can handle the output just fine.
+         *  - "useSpecialFloats": false or true
+         *  - If true, outputs non-finite floating point values in the following way:
+         *    NaN values as "NaN", positive infinity as "Infinity", and negative
+         *  infinity as "-Infinity".
+         *  - "precision": int
+         *  - Number of precision digits for formatting of real values.
+         *  - "precisionType": "significant"(default) or "decimal"
+         *  - Type of precision for formatting of real values.
+
+         *  You can examine 'settings_` yourself
+         *  to see the defaults. You can also write and read them just like any
+         *  JSON Value.
+         *  \sa setDefaults()
+         */
+        Json::Value settings_;
+
+        StreamWriterBuilder();
+
+        ~StreamWriterBuilder() override;
+
+        /**
+         * \throw std::exception if something goes wrong (e.g. invalid settings)
+         */
+        StreamWriter *newStreamWriter() const override;
+
+        /** \return true if 'settings' are legal and consistent;
+         *   otherwise, indicate bad settings via 'invalid'.
+         */
+        bool validate(Json::Value *invalid) const;
+
+        /** A simple way to update a specific setting.
+         */
+        Value &operator[](const String &key);
+
+        /** Called by ctor, but you can use this to reset settings_.
+         * \pre 'settings' != NULL (but Json::null is fine)
+         * \remark Defaults:
+         * \snippet src/lib_json/json_writer.cpp StreamWriterBuilderDefaults
+         */
+        static void setDefaults(Json::Value *settings);
+    };
+
+/** \brief Abstract class for writers.
+ * \deprecated Use StreamWriter. (And really, this is an implementation detail.)
+ */
+    class JSONCPP_DEPRECATED("Use StreamWriter instead") JSON_API Writer {
+    public:
+        virtual ~Writer();
+
+        virtual String write(const Value &root) = 0;
+    };
+
+/** \brief Outputs a Value in <a HREF="http://www.json.org">JSON</a> format
+ *without formatting (not human friendly).
+ *
+ * The JSON document is written in a single line. It is not intended for 'human'
+ *consumption,
+ * but may be useful to support feature such as RPC where bandwidth is limited.
+ * \sa Reader, Value
+ * \deprecated Use StreamWriterBuilder.
+ */
+#if defined(_MSC_VER)
+#pragma warning(push)
+#pragma warning(disable : 4996) // Deriving from deprecated class
+#endif
+
+    class JSONCPP_DEPRECATED("Use StreamWriterBuilder instead") JSON_API FastWriter
+            : public Writer {
+    public:
+        FastWriter();
+
+        ~FastWriter() override = default;
+
+        void enableYAMLCompatibility();
+
+        /** \brief Drop the "null" string from the writer's output for nullValues.
+         * Strictly speaking, this is not valid JSON. But when the output is being
+         * fed to a browser's JavaScript, it makes for smaller output and the
+         * browser can handle the output just fine.
+         */
+        void dropNullPlaceholders();
+
+        void omitEndingLineFeed();
+
+    public: // overridden from Writer
+        String write(const Value &root) override;
+
+    private:
+        void writeValue(const Value &value);
+
+        String document_;
+        bool yamlCompatibilityEnabled_{false};
+        bool dropNullPlaceholders_{false};
+        bool omitEndingLineFeed_{false};
+    };
+
+#if defined(_MSC_VER)
+#pragma warning(pop)
+#endif
+
+/** \brief Writes a Value in <a HREF="http://www.json.org">JSON</a> format in a
+ *human friendly way.
+ *
+ * The rules for line break and indent are as follow:
+ * - Object value:
+ *     - if empty then print {} without indent and line break
+ *     - if not empty the print '{', line break & indent, print one value per
+ *line
+ *       and then unindent and line break and print '}'.
+ * - Array value:
+ *     - if empty then print [] without indent and line break
+ *     - if the array contains no object value, empty array or some other value
+ *types,
+ *       and all the values fit on one lines, then print the array on a single
+ *line.
+ *     - otherwise, it the values do not fit on one line, or the array contains
+ *       object or non empty array, then print one value per line.
+ *
+ * If the Value have comments then they are outputed according to their
+ *#CommentPlacement.
+ *
+ * \sa Reader, Value, Value::setComment()
+ * \deprecated Use StreamWriterBuilder.
+ */
+#if defined(_MSC_VER)
+#pragma warning(push)
+#pragma warning(disable : 4996) // Deriving from deprecated class
+#endif
+
+    class JSONCPP_DEPRECATED("Use StreamWriterBuilder instead") JSON_API
+    StyledWriter : public Writer {
+    public:
+        StyledWriter();
+
+        ~StyledWriter() override = default;
+
+    public: // overridden from Writer
+        /** \brief Serialize a Value in <a HREF="http://www.json.org">JSON</a> format.
+         * \param root Value to serialize.
+         * \return String containing the JSON document that represents the root value.
+         */
+        String write(const Value &root) override;
+
+    private:
+        void writeValue(const Value &value);
+
+        void writeArrayValue(const Value &value);
+
+        bool isMultilineArray(const Value &value);
+
+        void pushValue(const String &value);
+
+        void writeIndent();
+
+        void writeWithIndent(const String &value);
+
+        void indent();
+
+        void unindent();
+
+        void writeCommentBeforeValue(const Value &root);
+
+        void writeCommentAfterValueOnSameLine(const Value &root);
+
+        static bool hasCommentForValue(const Value &value);
+
+        static String normalizeEOL(const String &text);
+
+        using ChildValues = std::vector<String>;
+
+        ChildValues childValues_;
+        String document_;
+        String indentString_;
+        unsigned int rightMargin_{74};
+        unsigned int indentSize_{3};
+        bool addChildValues_{false};
+    };
+
+#if defined(_MSC_VER)
+#pragma warning(pop)
+#endif
+
+/** \brief Writes a Value in <a HREF="http://www.json.org">JSON</a> format in a
+ human friendly way,
+     to a stream rather than to a string.
+ *
+ * The rules for line break and indent are as follow:
+ * - Object value:
+ *     - if empty then print {} without indent and line break
+ *     - if not empty the print '{', line break & indent, print one value per
+ line
+ *       and then unindent and line break and print '}'.
+ * - Array value:
+ *     - if empty then print [] without indent and line break
+ *     - if the array contains no object value, empty array or some other value
+ types,
+ *       and all the values fit on one lines, then print the array on a single
+ line.
+ *     - otherwise, it the values do not fit on one line, or the array contains
+ *       object or non empty array, then print one value per line.
+ *
+ * If the Value have comments then they are outputed according to their
+ #CommentPlacement.
+ *
+ * \sa Reader, Value, Value::setComment()
+ * \deprecated Use StreamWriterBuilder.
+ */
+#if defined(_MSC_VER)
+#pragma warning(push)
+#pragma warning(disable : 4996) // Deriving from deprecated class
+#endif
+
+    class JSONCPP_DEPRECATED("Use StreamWriterBuilder instead") JSON_API
+    StyledStreamWriter {
+    public:
+        /**
+         * \param indentation Each level will be indented by this amount extra.
+         */
+        StyledStreamWriter(String indentation = "\t");
+
+        ~StyledStreamWriter() = default;
+
+    public:
+        /** \brief Serialize a Value in <a HREF="http://www.json.org">JSON</a> format.
+         * \param out Stream to write to. (Can be ostringstream, e.g.)
+         * \param root Value to serialize.
+         * \note There is no point in deriving from Writer, since write() should not
+         * return a value.
+         */
+        void write(OStream &out, const Value &root);
+
+    private:
+        void writeValue(const Value &value);
+
+        void writeArrayValue(const Value &value);
+
+        bool isMultilineArray(const Value &value);
+
+        void pushValue(const String &value);
+
+        void writeIndent();
+
+        void writeWithIndent(const String &value);
+
+        void indent();
+
+        void unindent();
+
+        void writeCommentBeforeValue(const Value &root);
+
+        void writeCommentAfterValueOnSameLine(const Value &root);
+
+        static bool hasCommentForValue(const Value &value);
+
+        static String normalizeEOL(const String &text);
+
+        using ChildValues = std::vector<String>;
+
+        ChildValues childValues_;
+        OStream *document_;
+        String indentString_;
+        unsigned int rightMargin_{74};
+        String indentation_;
+        bool addChildValues_: 1;
+        bool indented_: 1;
+    };
+
+#if defined(_MSC_VER)
+#pragma warning(pop)
+#endif
+
+#if defined(JSON_HAS_INT64)
+
+    String JSON_API valueToString(Int value);
+
+    String JSON_API valueToString(UInt value);
+
+#endif // if defined(JSON_HAS_INT64)
+
+    String JSON_API valueToString(LargestInt value);
+
+    String JSON_API valueToString(LargestUInt value);
+
+    String JSON_API valueToString(
+            double value, unsigned int precision = Value::defaultRealPrecision,
+            PrecisionType precisionType = PrecisionType::significantDigits);
+
+    String JSON_API valueToString(bool value);
+
+    String JSON_API valueToQuotedString(const char *value);
+
+/// \brief Output using the StyledStreamWriter.
+/// \see Json::operator>>()
+    JSON_API OStream &operator<<(OStream &, const Value &root);
+
+} // namespace Json
+
+#pragma pack(pop)
+
+#if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+#pragma warning(pop)
+#endif // if defined(JSONCPP_DISABLE_DLL_INTERFACE_WARNING)
+
+#endif // JSON_WRITER_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/writer.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: include/json/assertions.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef JSON_ASSERTIONS_H_INCLUDED
+#define JSON_ASSERTIONS_H_INCLUDED
+
+#include <cstdlib>
+#include <sstream>
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include "config.h"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+/** It should not be possible for a maliciously designed file to
+ *  cause an abort() or seg-fault, so these macros are used only
+ *  for pre-condition violations and internal logic errors.
+ */
+#if JSON_USE_EXCEPTION
+
+// @todo <= add detail about condition in exception
+#define JSON_ASSERT(condition)                                                 \
+  do {                                                                         \
+    if (!(condition)) {                                                        \
+      Json::throwLogicError("assert json failed");                             \
+    }                                                                          \
+  } while (0)
+
+#define JSON_FAIL_MESSAGE(message)                                             \
+  do {                                                                         \
+    OStringStream oss;                                                         \
+    oss << message;                                                            \
+    Json::throwLogicError(oss.str());                                          \
+    abort();                                                                   \
+  } while (0)
+
+#else // JSON_USE_EXCEPTION
+
+#define JSON_ASSERT(condition) assert(condition)
+
+// The call to assert() will show the failure message in debug builds. In
+// release builds we abort, for a core-dump or debugger.
+#define JSON_FAIL_MESSAGE(message)                                             \
+  {                                                                            \
+    OStringStream oss;                                                         \
+    oss << message;                                                            \
+    assert(false && oss.str().c_str());                                        \
+    abort();                                                                   \
+  }
+
+#endif
+
+#define JSON_ASSERT_MESSAGE(condition, message)                                \
+  do {                                                                         \
+    if (!(condition)) {                                                        \
+      JSON_FAIL_MESSAGE(message);                                              \
+    }                                                                          \
+  } while (0)
+
+#endif // JSON_ASSERTIONS_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: include/json/assertions.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+#endif //ifndef JSON_AMALGAMATED_H_INCLUDED

+ 5448 - 0
library_core/src/main/cpp/external/json/jsoncpp.cpp

@@ -0,0 +1,5448 @@
+/// Json-cpp amalgamated source (http://jsoncpp.sourceforge.net/).
+/// It is intended to be used with #include "json/json.h"
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: LICENSE
+// //////////////////////////////////////////////////////////////////////
+
+/*
+The JsonCpp library's source code, including accompanying documentation, 
+tests and demonstration applications, are licensed under the following
+conditions...
+
+Baptiste Lepilleur and The JsonCpp Authors explicitly disclaim copyright in all 
+jurisdictions which recognize such a disclaimer. In such jurisdictions, 
+this software is released into the Public Domain.
+
+In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
+2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur and
+The JsonCpp Authors, and is released under the terms of the MIT License (see below).
+
+In jurisdictions which recognize Public Domain property, the user of this 
+software may choose to accept it either as 1) Public Domain, 2) under the 
+conditions of the MIT License (see below), or 3) under the terms of dual 
+Public Domain/MIT License conditions described here, as they choose.
+
+The MIT License is about as close to Public Domain as a license can get, and is
+described in clear, concise terms at:
+
+   http://en.wikipedia.org/wiki/MIT_License
+   
+The full text of the MIT License follows:
+
+========================================================================
+Copyright (c) 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+
+Permission is hereby granted, free of charge, to any person
+obtaining a copy of this software and associated documentation
+files (the "Software"), to deal in the Software without
+restriction, including without limitation the rights to use, copy,
+modify, merge, publish, distribute, sublicense, and/or sell copies
+of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
+BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
+ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
+========================================================================
+(END LICENSE TEXT)
+
+The MIT license is compatible with both the GPL and commercial
+software, affording one all of the rights of Public Domain with the
+minor nuisance of being required to keep the above copyright notice
+and license text in the source code. Note also that by accepting the
+Public Domain "license" you can re-license your copy using whatever
+license you like.
+
+*/
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: LICENSE
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+#include "json.h"
+
+#ifndef JSON_IS_AMALGAMATION
+#error "Compile with -I PATH_TO_JSON_DIRECTORY"
+#endif
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: src/lib_json/json_tool.h
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#ifndef LIB_JSONCPP_JSON_TOOL_H_INCLUDED
+#define LIB_JSONCPP_JSON_TOOL_H_INCLUDED
+
+#if !defined(JSON_IS_AMALGAMATION)
+#include <json/config.h>
+#endif
+
+// Also support old flag NO_LOCALE_SUPPORT
+#ifdef NO_LOCALE_SUPPORT
+#define JSONCPP_NO_LOCALE_SUPPORT
+#endif
+
+#ifndef JSONCPP_NO_LOCALE_SUPPORT
+
+#include <clocale>
+
+#endif
+
+/* This header provides common string manipulation support, such as UTF-8,
+ * portable conversion from/to string...
+ *
+ * It is an internal header that must not be exposed.
+ */
+
+namespace Json {
+    static inline char getDecimalPoint() {
+#ifdef JSONCPP_NO_LOCALE_SUPPORT
+        return '\0';
+#else
+        struct lconv *lc = localeconv();
+        return lc ? *(lc->decimal_point) : '\0';
+#endif
+    }
+
+/// Converts a unicode code-point to UTF-8.
+    static inline String codePointToUTF8(unsigned int cp) {
+        String result;
+
+        // based on description from http://en.wikipedia.org/wiki/UTF-8
+
+        if (cp <= 0x7f) {
+            result.resize(1);
+            result[0] = static_cast<char>(cp);
+        } else if (cp <= 0x7FF) {
+            result.resize(2);
+            result[1] = static_cast<char>(0x80 | (0x3f & cp));
+            result[0] = static_cast<char>(0xC0 | (0x1f & (cp >> 6)));
+        } else if (cp <= 0xFFFF) {
+            result.resize(3);
+            result[2] = static_cast<char>(0x80 | (0x3f & cp));
+            result[1] = static_cast<char>(0x80 | (0x3f & (cp >> 6)));
+            result[0] = static_cast<char>(0xE0 | (0xf & (cp >> 12)));
+        } else if (cp <= 0x10FFFF) {
+            result.resize(4);
+            result[3] = static_cast<char>(0x80 | (0x3f & cp));
+            result[2] = static_cast<char>(0x80 | (0x3f & (cp >> 6)));
+            result[1] = static_cast<char>(0x80 | (0x3f & (cp >> 12)));
+            result[0] = static_cast<char>(0xF0 | (0x7 & (cp >> 18)));
+        }
+
+        return result;
+    }
+
+    enum {
+        /// Constant that specify the size of the buffer that must be passed to
+        /// uintToString.
+        uintToStringBufferSize = 3 * sizeof(LargestUInt) + 1
+    };
+
+// Defines a char buffer for use with uintToString().
+    using UIntToStringBuffer = char[uintToStringBufferSize];
+
+/** Converts an unsigned integer to string.
+ * @param value Unsigned integer to convert to string
+ * @param current Input/Output string buffer.
+ *        Must have at least uintToStringBufferSize chars free.
+ */
+    static inline void uintToString(LargestUInt value, char *&current) {
+        *--current = 0;
+        do {
+            *--current = static_cast<char>(value % 10U + static_cast<unsigned>('0'));
+            value /= 10;
+        } while (value != 0);
+    }
+
+/** Change ',' to '.' everywhere in buffer.
+ *
+ * We had a sophisticated way, but it did not work in WinCE.
+ * @see https://github.com/open-source-parsers/jsoncpp/pull/9
+ */
+    template<typename Iter>
+    Iter fixNumericLocale(Iter begin, Iter end) {
+        for (; begin != end; ++begin) {
+            if (*begin == ',') {
+                *begin = '.';
+            }
+        }
+        return begin;
+    }
+
+    template<typename Iter>
+    void fixNumericLocaleInput(Iter begin, Iter end) {
+        char decimalPoint = getDecimalPoint();
+        if (decimalPoint == '\0' || decimalPoint == '.') {
+            return;
+        }
+        for (; begin != end; ++begin) {
+            if (*begin == '.') {
+                *begin = decimalPoint;
+            }
+        }
+    }
+
+/**
+ * Return iterator that would be the new end of the range [begin,end), if we
+ * were to delete zeros in the end of string, but not the last zero before '.'.
+ */
+    template<typename Iter>
+    Iter fixZerosInTheEnd(Iter begin, Iter end) {
+        for (; begin != end; --end) {
+            if (*(end - 1) != '0') {
+                return end;
+            }
+            // Don't delete the last zero before the decimal point.
+            if (begin != (end - 1) && *(end - 2) == '.') {
+                return end;
+            }
+        }
+        return end;
+    }
+
+} // namespace Json
+
+#endif // LIB_JSONCPP_JSON_TOOL_H_INCLUDED
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: src/lib_json/json_tool.h
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: src/lib_json/json_reader.cpp
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2011 Baptiste Lepilleur and The JsonCpp Authors
+// Copyright (C) 2016 InfoTeCS JSC. All rights reserved.
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#if !defined(JSON_IS_AMALGAMATION)
+                                                                                                                        #include "json_tool.h"
+#include <json/assertions.h>
+#include <json/reader.h>
+#include <json/value.h>
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+#include <algorithm>
+#include <cassert>
+#include <cstring>
+#include <iostream>
+#include <istream>
+#include <limits>
+#include <memory>
+#include <set>
+#include <sstream>
+#include <utility>
+
+#include <cstdio>
+
+#if __cplusplus >= 201103L
+
+#if !defined(sscanf)
+#define sscanf std::sscanf
+#endif
+
+#endif //__cplusplus
+
+#if defined(_MSC_VER)
+                                                                                                                        #if !defined(_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES)
+#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1
+#endif //_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES
+#endif //_MSC_VER
+
+#if defined(_MSC_VER)
+                                                                                                                        // Disable warning about strdup being deprecated.
+#pragma warning(disable : 4996)
+#endif
+
+// Define JSONCPP_DEPRECATED_STACK_LIMIT as an appropriate integer at compile
+// time to change the stack limit
+#if !defined(JSONCPP_DEPRECATED_STACK_LIMIT)
+#define JSONCPP_DEPRECATED_STACK_LIMIT 1000
+#endif
+
+static size_t const stackLimit_g =
+        JSONCPP_DEPRECATED_STACK_LIMIT; // see readValue()
+
+namespace Json {
+
+#if __cplusplus >= 201103L || (defined(_CPPLIB_VER) && _CPPLIB_VER >= 520)
+    using CharReaderPtr = std::unique_ptr<CharReader>;
+#else
+    using CharReaderPtr = std::auto_ptr<CharReader>;
+#endif
+
+// Implementation of class Features
+// ////////////////////////////////
+
+    Features::Features() = default;
+
+    Features Features::all() { return {}; }
+
+    Features Features::strictMode() {
+        Features features;
+        features.allowComments_ = false;
+        features.strictRoot_ = true;
+        features.allowDroppedNullPlaceholders_ = false;
+        features.allowNumericKeys_ = false;
+        return features;
+    }
+
+// Implementation of class Reader
+// ////////////////////////////////
+
+    bool Reader::containsNewLine(Reader::Location begin, Reader::Location end) {
+        return std::any_of(begin, end, [](char b) { return b == '\n' || b == '\r'; });
+    }
+
+// Class Reader
+// //////////////////////////////////////////////////////////////////
+
+    Reader::Reader() : features_(Features::all()) {}
+
+    Reader::Reader(const Features &features) : features_(features) {}
+
+    bool Reader::parse(const std::string &document, Value &root,
+                       bool collectComments) {
+        document_.assign(document.begin(), document.end());
+        const char *begin = document_.c_str();
+        const char *end = begin + document_.length();
+        return parse(begin, end, root, collectComments);
+    }
+
+    bool Reader::parse(std::istream &is, Value &root, bool collectComments) {
+        // std::istream_iterator<char> begin(is);
+        // std::istream_iterator<char> end;
+        // Those would allow streamed input from a file, if parse() were a
+        // template function.
+
+        // Since String is reference-counted, this at least does not
+        // create an extra copy.
+        String doc;
+        std::getline(is, doc, static_cast<char> EOF);
+        return parse(doc.data(), doc.data() + doc.size(), root, collectComments);
+    }
+
+    bool Reader::parse(const char *beginDoc, const char *endDoc, Value &root,
+                       bool collectComments) {
+        if (!features_.allowComments_) {
+            collectComments = false;
+        }
+
+        begin_ = beginDoc;
+        end_ = endDoc;
+        collectComments_ = collectComments;
+        current_ = begin_;
+        lastValueEnd_ = nullptr;
+        lastValue_ = nullptr;
+        commentsBefore_.clear();
+        errors_.clear();
+        while (!nodes_.empty())
+            nodes_.pop();
+        nodes_.push(&root);
+
+        bool successful = readValue();
+        Token token;
+        skipCommentTokens(token);
+        if (collectComments_ && !commentsBefore_.empty())
+            root.setComment(commentsBefore_, commentAfter);
+        if (features_.strictRoot_) {
+            if (!root.isArray() && !root.isObject()) {
+                // Set error location to start of doc, ideally should be first token found
+                // in doc
+                token.type_ = tokenError;
+                token.start_ = beginDoc;
+                token.end_ = endDoc;
+                addError(
+                        "A valid JSON document must be either an array or an object value.",
+                        token);
+                return false;
+            }
+        }
+        return successful;
+    }
+
+    bool Reader::readValue() {
+        // readValue() may call itself only if it calls readObject() or ReadArray().
+        // These methods execute nodes_.push() just before and nodes_.pop)() just
+        // after calling readValue(). parse() executes one nodes_.push(), so > instead
+        // of >=.
+        if (nodes_.size() > stackLimit_g)
+            throwRuntimeError("Exceeded stackLimit in readValue().");
+
+        Token token;
+        skipCommentTokens(token);
+        bool successful = true;
+
+        if (collectComments_ && !commentsBefore_.empty()) {
+            currentValue().setComment(commentsBefore_, commentBefore);
+            commentsBefore_.clear();
+        }
+
+        switch (token.type_) {
+            case tokenObjectBegin:
+                successful = readObject(token);
+                currentValue().setOffsetLimit(current_ - begin_);
+                break;
+            case tokenArrayBegin:
+                successful = readArray(token);
+                currentValue().setOffsetLimit(current_ - begin_);
+                break;
+            case tokenNumber:
+                successful = decodeNumber(token);
+                break;
+            case tokenString:
+                successful = decodeString(token);
+                break;
+            case tokenTrue: {
+                Value v(true);
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenFalse: {
+                Value v(false);
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenNull: {
+                Value v;
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenArraySeparator:
+            case tokenObjectEnd:
+            case tokenArrayEnd:
+                if (features_.allowDroppedNullPlaceholders_) {
+                    // "Un-read" the current token and mark the current value as a null
+                    // token.
+                    current_--;
+                    Value v;
+                    currentValue().swapPayload(v);
+                    currentValue().setOffsetStart(current_ - begin_ - 1);
+                    currentValue().setOffsetLimit(current_ - begin_);
+                    break;
+                } // Else, fall through...
+            default:
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+                return addError("Syntax error: value, object or array expected.", token);
+        }
+
+        if (collectComments_) {
+            lastValueEnd_ = current_;
+            lastValue_ = &currentValue();
+        }
+
+        return successful;
+    }
+
+    void Reader::skipCommentTokens(Token &token) {
+        if (features_.allowComments_) {
+            do {
+                readToken(token);
+            } while (token.type_ == tokenComment);
+        } else {
+            readToken(token);
+        }
+    }
+
+    bool Reader::readToken(Token &token) {
+        skipSpaces();
+        token.start_ = current_;
+        Char c = getNextChar();
+        bool ok = true;
+        switch (c) {
+            case '{':
+                token.type_ = tokenObjectBegin;
+                break;
+            case '}':
+                token.type_ = tokenObjectEnd;
+                break;
+            case '[':
+                token.type_ = tokenArrayBegin;
+                break;
+            case ']':
+                token.type_ = tokenArrayEnd;
+                break;
+            case '"':
+                token.type_ = tokenString;
+                ok = readString();
+                break;
+            case '/':
+                token.type_ = tokenComment;
+                ok = readComment();
+                break;
+            case '0':
+            case '1':
+            case '2':
+            case '3':
+            case '4':
+            case '5':
+            case '6':
+            case '7':
+            case '8':
+            case '9':
+            case '-':
+                token.type_ = tokenNumber;
+                readNumber();
+                break;
+            case 't':
+                token.type_ = tokenTrue;
+                ok = match("rue", 3);
+                break;
+            case 'f':
+                token.type_ = tokenFalse;
+                ok = match("alse", 4);
+                break;
+            case 'n':
+                token.type_ = tokenNull;
+                ok = match("ull", 3);
+                break;
+            case ',':
+                token.type_ = tokenArraySeparator;
+                break;
+            case ':':
+                token.type_ = tokenMemberSeparator;
+                break;
+            case 0:
+                token.type_ = tokenEndOfStream;
+                break;
+            default:
+                ok = false;
+                break;
+        }
+        if (!ok)
+            token.type_ = tokenError;
+        token.end_ = current_;
+        return ok;
+    }
+
+    void Reader::skipSpaces() {
+        while (current_ != end_) {
+            Char c = *current_;
+            if (c == ' ' || c == '\t' || c == '\r' || c == '\n')
+                ++current_;
+            else
+                break;
+        }
+    }
+
+    bool Reader::match(const Char *pattern, int patternLength) {
+        if (end_ - current_ < patternLength)
+            return false;
+        int index = patternLength;
+        while (index--)
+            if (current_[index] != pattern[index])
+                return false;
+        current_ += patternLength;
+        return true;
+    }
+
+    bool Reader::readComment() {
+        Location commentBegin = current_ - 1;
+        Char c = getNextChar();
+        bool successful = false;
+        if (c == '*')
+            successful = readCStyleComment();
+        else if (c == '/')
+            successful = readCppStyleComment();
+        if (!successful)
+            return false;
+
+        if (collectComments_) {
+            CommentPlacement placement = commentBefore;
+            if (lastValueEnd_ && !containsNewLine(lastValueEnd_, commentBegin)) {
+                if (c != '*' || !containsNewLine(commentBegin, current_))
+                    placement = commentAfterOnSameLine;
+            }
+
+            addComment(commentBegin, current_, placement);
+        }
+        return true;
+    }
+
+    String Reader::normalizeEOL(Reader::Location begin, Reader::Location end) {
+        String normalized;
+        normalized.reserve(static_cast<size_t>(end - begin));
+        Reader::Location current = begin;
+        while (current != end) {
+            char c = *current++;
+            if (c == '\r') {
+                if (current != end && *current == '\n')
+                    // convert dos EOL
+                    ++current;
+                // convert Mac EOL
+                normalized += '\n';
+            } else {
+                normalized += c;
+            }
+        }
+        return normalized;
+    }
+
+    void Reader::addComment(Location begin, Location end,
+                            CommentPlacement placement) {
+        assert(collectComments_);
+        const String &normalized = normalizeEOL(begin, end);
+        if (placement == commentAfterOnSameLine) {
+            assert(lastValue_ != nullptr);
+            lastValue_->setComment(normalized, placement);
+        } else {
+            commentsBefore_ += normalized;
+        }
+    }
+
+    bool Reader::readCStyleComment() {
+        while ((current_ + 1) < end_) {
+            Char c = getNextChar();
+            if (c == '*' && *current_ == '/')
+                break;
+        }
+        return getNextChar() == '/';
+    }
+
+    bool Reader::readCppStyleComment() {
+        while (current_ != end_) {
+            Char c = getNextChar();
+            if (c == '\n')
+                break;
+            if (c == '\r') {
+                // Consume DOS EOL. It will be normalized in addComment.
+                if (current_ != end_ && *current_ == '\n')
+                    getNextChar();
+                // Break on Moc OS 9 EOL.
+                break;
+            }
+        }
+        return true;
+    }
+
+    void Reader::readNumber() {
+        Location p = current_;
+        char c = '0'; // stopgap for already consumed character
+        // integral part
+        while (c >= '0' && c <= '9')
+            c = (current_ = p) < end_ ? *p++ : '\0';
+        // fractional part
+        if (c == '.') {
+            c = (current_ = p) < end_ ? *p++ : '\0';
+            while (c >= '0' && c <= '9')
+                c = (current_ = p) < end_ ? *p++ : '\0';
+        }
+        // exponential part
+        if (c == 'e' || c == 'E') {
+            c = (current_ = p) < end_ ? *p++ : '\0';
+            if (c == '+' || c == '-')
+                c = (current_ = p) < end_ ? *p++ : '\0';
+            while (c >= '0' && c <= '9')
+                c = (current_ = p) < end_ ? *p++ : '\0';
+        }
+    }
+
+    bool Reader::readString() {
+        Char c = '\0';
+        while (current_ != end_) {
+            c = getNextChar();
+            if (c == '\\')
+                getNextChar();
+            else if (c == '"')
+                break;
+        }
+        return c == '"';
+    }
+
+    bool Reader::readObject(Token &token) {
+        Token tokenName;
+        String name;
+        Value init(objectValue);
+        currentValue().swapPayload(init);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        while (readToken(tokenName)) {
+            bool initialTokenOk = true;
+            while (tokenName.type_ == tokenComment && initialTokenOk)
+                initialTokenOk = readToken(tokenName);
+            if (!initialTokenOk)
+                break;
+            if (tokenName.type_ == tokenObjectEnd && name.empty()) // empty object
+                return true;
+            name.clear();
+            if (tokenName.type_ == tokenString) {
+                if (!decodeString(tokenName, name))
+                    return recoverFromError(tokenObjectEnd);
+            } else if (tokenName.type_ == tokenNumber && features_.allowNumericKeys_) {
+                Value numberName;
+                if (!decodeNumber(tokenName, numberName))
+                    return recoverFromError(tokenObjectEnd);
+                name = numberName.asString();
+            } else {
+                break;
+            }
+
+            Token colon;
+            if (!readToken(colon) || colon.type_ != tokenMemberSeparator) {
+                return addErrorAndRecover("Missing ':' after object member name", colon,
+                                          tokenObjectEnd);
+            }
+            Value &value = currentValue()[name];
+            nodes_.push(&value);
+            bool ok = readValue();
+            nodes_.pop();
+            if (!ok) // error already put
+                return recoverFromError(tokenObjectEnd);
+
+            Token comma;
+            if (!readToken(comma) ||
+                (comma.type_ != tokenObjectEnd && comma.type_ != tokenArraySeparator &&
+                 comma.type_ != tokenComment)) {
+                return addErrorAndRecover("Missing ',' or '}' in object declaration",
+                                          comma, tokenObjectEnd);
+            }
+            bool finalizeTokenOk = true;
+            while (comma.type_ == tokenComment && finalizeTokenOk)
+                finalizeTokenOk = readToken(comma);
+            if (comma.type_ == tokenObjectEnd)
+                return true;
+        }
+        return addErrorAndRecover("Missing '}' or object member name", tokenName,
+                                  tokenObjectEnd);
+    }
+
+    bool Reader::readArray(Token &token) {
+        Value init(arrayValue);
+        currentValue().swapPayload(init);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        skipSpaces();
+        if (current_ != end_ && *current_ == ']') // empty array
+        {
+            Token endArray;
+            readToken(endArray);
+            return true;
+        }
+        int index = 0;
+        for (;;) {
+            Value &value = currentValue()[index++];
+            nodes_.push(&value);
+            bool ok = readValue();
+            nodes_.pop();
+            if (!ok) // error already put
+                return recoverFromError(tokenArrayEnd);
+
+            Token currentToken;
+            // Accept Comment after last item in the array.
+            ok = readToken(currentToken);
+            while (currentToken.type_ == tokenComment && ok) {
+                ok = readToken(currentToken);
+            }
+            bool badTokenType = (currentToken.type_ != tokenArraySeparator &&
+                                 currentToken.type_ != tokenArrayEnd);
+            if (!ok || badTokenType) {
+                return addErrorAndRecover("Missing ',' or ']' in array declaration",
+                                          currentToken, tokenArrayEnd);
+            }
+            if (currentToken.type_ == tokenArrayEnd)
+                break;
+        }
+        return true;
+    }
+
+    bool Reader::decodeNumber(Token &token) {
+        Value decoded;
+        if (!decodeNumber(token, decoded))
+            return false;
+        currentValue().swapPayload(decoded);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        currentValue().setOffsetLimit(token.end_ - begin_);
+        return true;
+    }
+
+    bool Reader::decodeNumber(Token &token, Value &decoded) {
+        // Attempts to parse the number as an integer. If the number is
+        // larger than the maximum supported value of an integer then
+        // we decode the number as a double.
+        Location current = token.start_;
+        bool isNegative = *current == '-';
+        if (isNegative)
+            ++current;
+        // TODO: Help the compiler do the div and mod at compile time or get rid of
+        // them.
+        Value::LargestUInt maxIntegerValue =
+                isNegative ? Value::LargestUInt(Value::maxLargestInt) + 1
+                           : Value::maxLargestUInt;
+        Value::LargestUInt threshold = maxIntegerValue / 10;
+        Value::LargestUInt value = 0;
+        while (current < token.end_) {
+            Char c = *current++;
+            if (c < '0' || c > '9')
+                return decodeDouble(token, decoded);
+            auto digit(static_cast<Value::UInt>(c - '0'));
+            if (value >= threshold) {
+                // We've hit or exceeded the max value divided by 10 (rounded down). If
+                // a) we've only just touched the limit, b) this is the last digit, and
+                // c) it's small enough to fit in that rounding delta, we're okay.
+                // Otherwise treat this number as a double to avoid overflow.
+                if (value > threshold || current != token.end_ ||
+                    digit > maxIntegerValue % 10) {
+                    return decodeDouble(token, decoded);
+                }
+            }
+            value = value * 10 + digit;
+        }
+        if (isNegative && value == maxIntegerValue)
+            decoded = Value::minLargestInt;
+        else if (isNegative)
+            decoded = -Value::LargestInt(value);
+        else if (value <= Value::LargestUInt(Value::maxInt))
+            decoded = Value::LargestInt(value);
+        else
+            decoded = value;
+        return true;
+    }
+
+    bool Reader::decodeDouble(Token &token) {
+        Value decoded;
+        if (!decodeDouble(token, decoded))
+            return false;
+        currentValue().swapPayload(decoded);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        currentValue().setOffsetLimit(token.end_ - begin_);
+        return true;
+    }
+
+    bool Reader::decodeDouble(Token &token, Value &decoded) {
+        double value = 0;
+        String buffer(token.start_, token.end_);
+        IStringStream is(buffer);
+        if (!(is >> value))
+            return addError(
+                    "'" + String(token.start_, token.end_) + "' is not a number.", token);
+        decoded = value;
+        return true;
+    }
+
+    bool Reader::decodeString(Token &token) {
+        String decoded_string;
+        if (!decodeString(token, decoded_string))
+            return false;
+        Value decoded(decoded_string);
+        currentValue().swapPayload(decoded);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        currentValue().setOffsetLimit(token.end_ - begin_);
+        return true;
+    }
+
+    bool Reader::decodeString(Token &token, String &decoded) {
+        decoded.reserve(static_cast<size_t>(token.end_ - token.start_ - 2));
+        Location current = token.start_ + 1; // skip '"'
+        Location end = token.end_ - 1;       // do not include '"'
+        while (current != end) {
+            Char c = *current++;
+            if (c == '"')
+                break;
+            if (c == '\\') {
+                if (current == end)
+                    return addError("Empty escape sequence in string", token, current);
+                Char escape = *current++;
+                switch (escape) {
+                    case '"':
+                        decoded += '"';
+                        break;
+                    case '/':
+                        decoded += '/';
+                        break;
+                    case '\\':
+                        decoded += '\\';
+                        break;
+                    case 'b':
+                        decoded += '\b';
+                        break;
+                    case 'f':
+                        decoded += '\f';
+                        break;
+                    case 'n':
+                        decoded += '\n';
+                        break;
+                    case 'r':
+                        decoded += '\r';
+                        break;
+                    case 't':
+                        decoded += '\t';
+                        break;
+                    case 'u': {
+                        unsigned int unicode;
+                        if (!decodeUnicodeCodePoint(token, current, end, unicode))
+                            return false;
+                        decoded += codePointToUTF8(unicode);
+                    }
+                        break;
+                    default:
+                        return addError("Bad escape sequence in string", token, current);
+                }
+            } else {
+                decoded += c;
+            }
+        }
+        return true;
+    }
+
+    bool Reader::decodeUnicodeCodePoint(Token &token, Location &current,
+                                        Location end, unsigned int &unicode) {
+
+        if (!decodeUnicodeEscapeSequence(token, current, end, unicode))
+            return false;
+        if (unicode >= 0xD800 && unicode <= 0xDBFF) {
+            // surrogate pairs
+            if (end - current < 6)
+                return addError(
+                        "additional six characters expected to parse unicode surrogate pair.",
+                        token, current);
+            if (*(current++) == '\\' && *(current++) == 'u') {
+                unsigned int surrogatePair;
+                if (decodeUnicodeEscapeSequence(token, current, end, surrogatePair)) {
+                    unicode = 0x10000 + ((unicode & 0x3FF) << 10) + (surrogatePair & 0x3FF);
+                } else
+                    return false;
+            } else
+                return addError("expecting another \\u token to begin the second half of "
+                                "a unicode surrogate pair",
+                                token, current);
+        }
+        return true;
+    }
+
+    bool Reader::decodeUnicodeEscapeSequence(Token &token, Location &current,
+                                             Location end,
+                                             unsigned int &ret_unicode) {
+        if (end - current < 4)
+            return addError(
+                    "Bad unicode escape sequence in string: four digits expected.", token,
+                    current);
+        int unicode = 0;
+        for (int index = 0; index < 4; ++index) {
+            Char c = *current++;
+            unicode *= 16;
+            if (c >= '0' && c <= '9')
+                unicode += c - '0';
+            else if (c >= 'a' && c <= 'f')
+                unicode += c - 'a' + 10;
+            else if (c >= 'A' && c <= 'F')
+                unicode += c - 'A' + 10;
+            else
+                return addError(
+                        "Bad unicode escape sequence in string: hexadecimal digit expected.",
+                        token, current);
+        }
+        ret_unicode = static_cast<unsigned int>(unicode);
+        return true;
+    }
+
+    bool Reader::addError(const String &message, Token &token, Location extra) {
+        ErrorInfo info;
+        info.token_ = token;
+        info.message_ = message;
+        info.extra_ = extra;
+        errors_.push_back(info);
+        return false;
+    }
+
+    bool Reader::recoverFromError(TokenType skipUntilToken) {
+        size_t const errorCount = errors_.size();
+        Token skip;
+        for (;;) {
+            if (!readToken(skip))
+                errors_.resize(errorCount); // discard errors caused by recovery
+            if (skip.type_ == skipUntilToken || skip.type_ == tokenEndOfStream)
+                break;
+        }
+        errors_.resize(errorCount);
+        return false;
+    }
+
+    bool Reader::addErrorAndRecover(const String &message, Token &token,
+                                    TokenType skipUntilToken) {
+        addError(message, token);
+        return recoverFromError(skipUntilToken);
+    }
+
+    Value &Reader::currentValue() { return *(nodes_.top()); }
+
+    Reader::Char Reader::getNextChar() {
+        if (current_ == end_)
+            return 0;
+        return *current_++;
+    }
+
+    void Reader::getLocationLineAndColumn(Location location, int &line,
+                                          int &column) const {
+        Location current = begin_;
+        Location lastLineStart = current;
+        line = 0;
+        while (current < location && current != end_) {
+            Char c = *current++;
+            if (c == '\r') {
+                if (*current == '\n')
+                    ++current;
+                lastLineStart = current;
+                ++line;
+            } else if (c == '\n') {
+                lastLineStart = current;
+                ++line;
+            }
+        }
+        // column & line start at 1
+        column = int(location - lastLineStart) + 1;
+        ++line;
+    }
+
+    String Reader::getLocationLineAndColumn(Location location) const {
+        int line, column;
+        getLocationLineAndColumn(location, line, column);
+        char buffer[18 + 16 + 16 + 1];
+        jsoncpp_snprintf(buffer, sizeof(buffer), "Line %d, Column %d", line, column);
+        return buffer;
+    }
+
+// Deprecated. Preserved for backward compatibility
+    String Reader::getFormatedErrorMessages() const {
+        return getFormattedErrorMessages();
+    }
+
+    String Reader::getFormattedErrorMessages() const {
+        String formattedMessage;
+        for (const auto &error : errors_) {
+            formattedMessage +=
+                    "* " + getLocationLineAndColumn(error.token_.start_) + "\n";
+            formattedMessage += "  " + error.message_ + "\n";
+            if (error.extra_)
+                formattedMessage +=
+                        "See " + getLocationLineAndColumn(error.extra_) + " for detail.\n";
+        }
+        return formattedMessage;
+    }
+
+    std::vector<Reader::StructuredError> Reader::getStructuredErrors() const {
+        std::vector<Reader::StructuredError> allErrors;
+        for (const auto &error : errors_) {
+            Reader::StructuredError structured;
+            structured.offset_start = error.token_.start_ - begin_;
+            structured.offset_limit = error.token_.end_ - begin_;
+            structured.message = error.message_;
+            allErrors.push_back(structured);
+        }
+        return allErrors;
+    }
+
+    bool Reader::pushError(const Value &value, const String &message) {
+        ptrdiff_t const length = end_ - begin_;
+        if (value.getOffsetStart() > length || value.getOffsetLimit() > length)
+            return false;
+        Token token;
+        token.type_ = tokenError;
+        token.start_ = begin_ + value.getOffsetStart();
+        token.end_ = begin_ + value.getOffsetLimit();
+        ErrorInfo info;
+        info.token_ = token;
+        info.message_ = message;
+        info.extra_ = nullptr;
+        errors_.push_back(info);
+        return true;
+    }
+
+    bool Reader::pushError(const Value &value, const String &message,
+                           const Value &extra) {
+        ptrdiff_t const length = end_ - begin_;
+        if (value.getOffsetStart() > length || value.getOffsetLimit() > length ||
+            extra.getOffsetLimit() > length)
+            return false;
+        Token token;
+        token.type_ = tokenError;
+        token.start_ = begin_ + value.getOffsetStart();
+        token.end_ = begin_ + value.getOffsetLimit();
+        ErrorInfo info;
+        info.token_ = token;
+        info.message_ = message;
+        info.extra_ = begin_ + extra.getOffsetStart();
+        errors_.push_back(info);
+        return true;
+    }
+
+    bool Reader::good() const { return errors_.empty(); }
+
+// Originally copied from the Features class (now deprecated), used internally
+// for features implementation.
+    class OurFeatures {
+    public:
+        static OurFeatures all();
+
+        bool allowComments_;
+        bool allowTrailingCommas_;
+        bool strictRoot_;
+        bool allowDroppedNullPlaceholders_;
+        bool allowNumericKeys_;
+        bool allowSingleQuotes_;
+        bool failIfExtra_;
+        bool rejectDupKeys_;
+        bool allowSpecialFloats_;
+        bool skipBom_;
+        size_t stackLimit_;
+    }; // OurFeatures
+
+    OurFeatures OurFeatures::all() { return {}; }
+
+// Implementation of class Reader
+// ////////////////////////////////
+
+// Originally copied from the Reader class (now deprecated), used internally
+// for implementing JSON reading.
+    class OurReader {
+    public:
+        using Char = char;
+        using Location = const Char *;
+        struct StructuredError {
+            ptrdiff_t offset_start;
+            ptrdiff_t offset_limit;
+            String message;
+        };
+
+        explicit OurReader(OurFeatures const &features);
+
+        bool parse(const char *beginDoc, const char *endDoc, Value &root,
+                   bool collectComments = true);
+
+        String getFormattedErrorMessages() const;
+
+        std::vector<StructuredError> getStructuredErrors() const;
+
+    private:
+        OurReader(OurReader const &);      // no impl
+        void operator=(OurReader const &); // no impl
+
+        enum TokenType {
+            tokenEndOfStream = 0,
+            tokenObjectBegin,
+            tokenObjectEnd,
+            tokenArrayBegin,
+            tokenArrayEnd,
+            tokenString,
+            tokenNumber,
+            tokenTrue,
+            tokenFalse,
+            tokenNull,
+            tokenNaN,
+            tokenPosInf,
+            tokenNegInf,
+            tokenArraySeparator,
+            tokenMemberSeparator,
+            tokenComment,
+            tokenError
+        };
+
+        class Token {
+        public:
+            TokenType type_;
+            Location start_;
+            Location end_;
+        };
+
+        class ErrorInfo {
+        public:
+            Token token_;
+            String message_;
+            Location extra_;
+        };
+
+        using Errors = std::deque<ErrorInfo>;
+
+        bool readToken(Token &token);
+
+        void skipSpaces();
+
+        void skipBom(bool skipBom);
+
+        bool match(const Char *pattern, int patternLength);
+
+        bool readComment();
+
+        bool readCStyleComment(bool *containsNewLineResult);
+
+        bool readCppStyleComment();
+
+        bool readString();
+
+        bool readStringSingleQuote();
+
+        bool readNumber(bool checkInf);
+
+        bool readValue();
+
+        bool readObject(Token &token);
+
+        bool readArray(Token &token);
+
+        bool decodeNumber(Token &token);
+
+        bool decodeNumber(Token &token, Value &decoded);
+
+        bool decodeString(Token &token);
+
+        bool decodeString(Token &token, String &decoded);
+
+        bool decodeDouble(Token &token);
+
+        bool decodeDouble(Token &token, Value &decoded);
+
+        bool decodeUnicodeCodePoint(Token &token, Location &current, Location end,
+                                    unsigned int &unicode);
+
+        bool decodeUnicodeEscapeSequence(Token &token, Location &current,
+                                         Location end, unsigned int &unicode);
+
+        bool addError(const String &message, Token &token, Location extra = nullptr);
+
+        bool recoverFromError(TokenType skipUntilToken);
+
+        bool addErrorAndRecover(const String &message, Token &token,
+                                TokenType skipUntilToken);
+
+        void skipUntilSpace();
+
+        Value &currentValue();
+
+        Char getNextChar();
+
+        void getLocationLineAndColumn(Location location, int &line,
+                                      int &column) const;
+
+        String getLocationLineAndColumn(Location location) const;
+
+        void addComment(Location begin, Location end, CommentPlacement placement);
+
+        void skipCommentTokens(Token &token);
+
+        static String normalizeEOL(Location begin, Location end);
+
+        static bool containsNewLine(Location begin, Location end);
+
+        using Nodes = std::stack<Value *>;
+
+        Nodes nodes_{};
+        Errors errors_{};
+        String document_{};
+        Location begin_ = nullptr;
+        Location end_ = nullptr;
+        Location current_ = nullptr;
+        Location lastValueEnd_ = nullptr;
+        Value *lastValue_ = nullptr;
+        bool lastValueHasAComment_ = false;
+        String commentsBefore_{};
+
+        OurFeatures const features_;
+        bool collectComments_ = false;
+    }; // OurReader
+
+// complete copy of Read impl, for OurReader
+
+    bool OurReader::containsNewLine(OurReader::Location begin,
+                                    OurReader::Location end) {
+        return std::any_of(begin, end, [](char b) { return b == '\n' || b == '\r'; });
+    }
+
+    OurReader::OurReader(OurFeatures const &features) : features_(features) {}
+
+    bool OurReader::parse(const char *beginDoc, const char *endDoc, Value &root,
+                          bool collectComments) {
+        if (!features_.allowComments_) {
+            collectComments = false;
+        }
+
+        begin_ = beginDoc;
+        end_ = endDoc;
+        collectComments_ = collectComments;
+        current_ = begin_;
+        lastValueEnd_ = nullptr;
+        lastValue_ = nullptr;
+        commentsBefore_.clear();
+        errors_.clear();
+        while (!nodes_.empty())
+            nodes_.pop();
+        nodes_.push(&root);
+
+        // skip byte order mark if it exists at the beginning of the UTF-8 text.
+        skipBom(features_.skipBom_);
+        bool successful = readValue();
+        nodes_.pop();
+        Token token;
+        skipCommentTokens(token);
+        if (features_.failIfExtra_ && (token.type_ != tokenEndOfStream)) {
+            addError("Extra non-whitespace after JSON value.", token);
+            return false;
+        }
+        if (collectComments_ && !commentsBefore_.empty())
+            root.setComment(commentsBefore_, commentAfter);
+        if (features_.strictRoot_) {
+            if (!root.isArray() && !root.isObject()) {
+                // Set error location to start of doc, ideally should be first token found
+                // in doc
+                token.type_ = tokenError;
+                token.start_ = beginDoc;
+                token.end_ = endDoc;
+                addError(
+                        "A valid JSON document must be either an array or an object value.",
+                        token);
+                return false;
+            }
+        }
+        return successful;
+    }
+
+    bool OurReader::readValue() {
+        //  To preserve the old behaviour we cast size_t to int.
+        if (nodes_.size() > features_.stackLimit_)
+            throwRuntimeError("Exceeded stackLimit in readValue().");
+        Token token;
+        skipCommentTokens(token);
+        bool successful = true;
+
+        if (collectComments_ && !commentsBefore_.empty()) {
+            currentValue().setComment(commentsBefore_, commentBefore);
+            commentsBefore_.clear();
+        }
+
+        switch (token.type_) {
+            case tokenObjectBegin:
+                successful = readObject(token);
+                currentValue().setOffsetLimit(current_ - begin_);
+                break;
+            case tokenArrayBegin:
+                successful = readArray(token);
+                currentValue().setOffsetLimit(current_ - begin_);
+                break;
+            case tokenNumber:
+                successful = decodeNumber(token);
+                break;
+            case tokenString:
+                successful = decodeString(token);
+                break;
+            case tokenTrue: {
+                Value v(true);
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenFalse: {
+                Value v(false);
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenNull: {
+                Value v;
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenNaN: {
+                Value v(std::numeric_limits<double>::quiet_NaN());
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenPosInf: {
+                Value v(std::numeric_limits<double>::infinity());
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenNegInf: {
+                Value v(-std::numeric_limits<double>::infinity());
+                currentValue().swapPayload(v);
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+            }
+                break;
+            case tokenArraySeparator:
+            case tokenObjectEnd:
+            case tokenArrayEnd:
+                if (features_.allowDroppedNullPlaceholders_) {
+                    // "Un-read" the current token and mark the current value as a null
+                    // token.
+                    current_--;
+                    Value v;
+                    currentValue().swapPayload(v);
+                    currentValue().setOffsetStart(current_ - begin_ - 1);
+                    currentValue().setOffsetLimit(current_ - begin_);
+                    break;
+                } // else, fall through ...
+            default:
+                currentValue().setOffsetStart(token.start_ - begin_);
+                currentValue().setOffsetLimit(token.end_ - begin_);
+                return addError("Syntax error: value, object or array expected.", token);
+        }
+
+        if (collectComments_) {
+            lastValueEnd_ = current_;
+            lastValueHasAComment_ = false;
+            lastValue_ = &currentValue();
+        }
+
+        return successful;
+    }
+
+    void OurReader::skipCommentTokens(Token &token) {
+        if (features_.allowComments_) {
+            do {
+                readToken(token);
+            } while (token.type_ == tokenComment);
+        } else {
+            readToken(token);
+        }
+    }
+
+    bool OurReader::readToken(Token &token) {
+        skipSpaces();
+        token.start_ = current_;
+        Char c = getNextChar();
+        bool ok = true;
+        switch (c) {
+            case '{':
+                token.type_ = tokenObjectBegin;
+                break;
+            case '}':
+                token.type_ = tokenObjectEnd;
+                break;
+            case '[':
+                token.type_ = tokenArrayBegin;
+                break;
+            case ']':
+                token.type_ = tokenArrayEnd;
+                break;
+            case '"':
+                token.type_ = tokenString;
+                ok = readString();
+                break;
+            case '\'':
+                if (features_.allowSingleQuotes_) {
+                    token.type_ = tokenString;
+                    ok = readStringSingleQuote();
+                } else {
+                    // If we don't allow single quotes, this is a failure case.
+                    ok = false;
+                }
+                break;
+            case '/':
+                token.type_ = tokenComment;
+                ok = readComment();
+                break;
+            case '0':
+            case '1':
+            case '2':
+            case '3':
+            case '4':
+            case '5':
+            case '6':
+            case '7':
+            case '8':
+            case '9':
+                token.type_ = tokenNumber;
+                readNumber(false);
+                break;
+            case '-':
+                if (readNumber(true)) {
+                    token.type_ = tokenNumber;
+                } else {
+                    token.type_ = tokenNegInf;
+                    ok = features_.allowSpecialFloats_ && match("nfinity", 7);
+                }
+                break;
+            case '+':
+                if (readNumber(true)) {
+                    token.type_ = tokenNumber;
+                } else {
+                    token.type_ = tokenPosInf;
+                    ok = features_.allowSpecialFloats_ && match("nfinity", 7);
+                }
+                break;
+            case 't':
+                token.type_ = tokenTrue;
+                ok = match("rue", 3);
+                break;
+            case 'f':
+                token.type_ = tokenFalse;
+                ok = match("alse", 4);
+                break;
+            case 'n':
+                token.type_ = tokenNull;
+                ok = match("ull", 3);
+                break;
+            case 'N':
+                if (features_.allowSpecialFloats_) {
+                    token.type_ = tokenNaN;
+                    ok = match("aN", 2);
+                } else {
+                    ok = false;
+                }
+                break;
+            case 'I':
+                if (features_.allowSpecialFloats_) {
+                    token.type_ = tokenPosInf;
+                    ok = match("nfinity", 7);
+                } else {
+                    ok = false;
+                }
+                break;
+            case ',':
+                token.type_ = tokenArraySeparator;
+                break;
+            case ':':
+                token.type_ = tokenMemberSeparator;
+                break;
+            case 0:
+                token.type_ = tokenEndOfStream;
+                break;
+            default:
+                ok = false;
+                break;
+        }
+        if (!ok)
+            token.type_ = tokenError;
+        token.end_ = current_;
+        return ok;
+    }
+
+    void OurReader::skipSpaces() {
+        while (current_ != end_) {
+            Char c = *current_;
+            if (c == ' ' || c == '\t' || c == '\r' || c == '\n')
+                ++current_;
+            else
+                break;
+        }
+    }
+
+    void OurReader::skipBom(bool skipBom) {
+        // The default behavior is to skip BOM.
+        if (skipBom) {
+            if ((end_ - begin_) >= 3 && strncmp(begin_, "\xEF\xBB\xBF", 3) == 0) {
+                begin_ += 3;
+                current_ = begin_;
+            }
+        }
+    }
+
+    bool OurReader::match(const Char *pattern, int patternLength) {
+        if (end_ - current_ < patternLength)
+            return false;
+        int index = patternLength;
+        while (index--)
+            if (current_[index] != pattern[index])
+                return false;
+        current_ += patternLength;
+        return true;
+    }
+
+    bool OurReader::readComment() {
+        const Location commentBegin = current_ - 1;
+        const Char c = getNextChar();
+        bool successful = false;
+        bool cStyleWithEmbeddedNewline = false;
+
+        const bool isCStyleComment = (c == '*');
+        const bool isCppStyleComment = (c == '/');
+        if (isCStyleComment) {
+            successful = readCStyleComment(&cStyleWithEmbeddedNewline);
+        } else if (isCppStyleComment) {
+            successful = readCppStyleComment();
+        }
+
+        if (!successful)
+            return false;
+
+        if (collectComments_) {
+            CommentPlacement placement = commentBefore;
+
+            if (!lastValueHasAComment_) {
+                if (lastValueEnd_ && !containsNewLine(lastValueEnd_, commentBegin)) {
+                    if (isCppStyleComment || !cStyleWithEmbeddedNewline) {
+                        placement = commentAfterOnSameLine;
+                        lastValueHasAComment_ = true;
+                    }
+                }
+            }
+
+            addComment(commentBegin, current_, placement);
+        }
+        return true;
+    }
+
+    String OurReader::normalizeEOL(OurReader::Location begin,
+                                   OurReader::Location end) {
+        String normalized;
+        normalized.reserve(static_cast<size_t>(end - begin));
+        OurReader::Location current = begin;
+        while (current != end) {
+            char c = *current++;
+            if (c == '\r') {
+                if (current != end && *current == '\n')
+                    // convert dos EOL
+                    ++current;
+                // convert Mac EOL
+                normalized += '\n';
+            } else {
+                normalized += c;
+            }
+        }
+        return normalized;
+    }
+
+    void OurReader::addComment(Location begin, Location end,
+                               CommentPlacement placement) {
+        assert(collectComments_);
+        const String &normalized = normalizeEOL(begin, end);
+        if (placement == commentAfterOnSameLine) {
+            assert(lastValue_ != nullptr);
+            lastValue_->setComment(normalized, placement);
+        } else {
+            commentsBefore_ += normalized;
+        }
+    }
+
+    bool OurReader::readCStyleComment(bool *containsNewLineResult) {
+        *containsNewLineResult = false;
+
+        while ((current_ + 1) < end_) {
+            Char c = getNextChar();
+            if (c == '*' && *current_ == '/')
+                break;
+            if (c == '\n')
+                *containsNewLineResult = true;
+        }
+
+        return getNextChar() == '/';
+    }
+
+    bool OurReader::readCppStyleComment() {
+        while (current_ != end_) {
+            Char c = getNextChar();
+            if (c == '\n')
+                break;
+            if (c == '\r') {
+                // Consume DOS EOL. It will be normalized in addComment.
+                if (current_ != end_ && *current_ == '\n')
+                    getNextChar();
+                // Break on Moc OS 9 EOL.
+                break;
+            }
+        }
+        return true;
+    }
+
+    bool OurReader::readNumber(bool checkInf) {
+        Location p = current_;
+        if (checkInf && p != end_ && *p == 'I') {
+            current_ = ++p;
+            return false;
+        }
+        char c = '0'; // stopgap for already consumed character
+        // integral part
+        while (c >= '0' && c <= '9')
+            c = (current_ = p) < end_ ? *p++ : '\0';
+        // fractional part
+        if (c == '.') {
+            c = (current_ = p) < end_ ? *p++ : '\0';
+            while (c >= '0' && c <= '9')
+                c = (current_ = p) < end_ ? *p++ : '\0';
+        }
+        // exponential part
+        if (c == 'e' || c == 'E') {
+            c = (current_ = p) < end_ ? *p++ : '\0';
+            if (c == '+' || c == '-')
+                c = (current_ = p) < end_ ? *p++ : '\0';
+            while (c >= '0' && c <= '9')
+                c = (current_ = p) < end_ ? *p++ : '\0';
+        }
+        return true;
+    }
+
+    bool OurReader::readString() {
+        Char c = 0;
+        while (current_ != end_) {
+            c = getNextChar();
+            if (c == '\\')
+                getNextChar();
+            else if (c == '"')
+                break;
+        }
+        return c == '"';
+    }
+
+    bool OurReader::readStringSingleQuote() {
+        Char c = 0;
+        while (current_ != end_) {
+            c = getNextChar();
+            if (c == '\\')
+                getNextChar();
+            else if (c == '\'')
+                break;
+        }
+        return c == '\'';
+    }
+
+    bool OurReader::readObject(Token &token) {
+        Token tokenName;
+        String name;
+        Value init(objectValue);
+        currentValue().swapPayload(init);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        while (readToken(tokenName)) {
+            bool initialTokenOk = true;
+            while (tokenName.type_ == tokenComment && initialTokenOk)
+                initialTokenOk = readToken(tokenName);
+            if (!initialTokenOk)
+                break;
+            if (tokenName.type_ == tokenObjectEnd &&
+                (name.empty() ||
+                 features_.allowTrailingCommas_)) // empty object or trailing comma
+                return true;
+            name.clear();
+            if (tokenName.type_ == tokenString) {
+                if (!decodeString(tokenName, name))
+                    return recoverFromError(tokenObjectEnd);
+            } else if (tokenName.type_ == tokenNumber && features_.allowNumericKeys_) {
+                Value numberName;
+                if (!decodeNumber(tokenName, numberName))
+                    return recoverFromError(tokenObjectEnd);
+                name = numberName.asString();
+            } else {
+                break;
+            }
+            if (name.length() >= (1U << 30))
+                throwRuntimeError("keylength >= 2^30");
+            if (features_.rejectDupKeys_ && currentValue().isMember(name)) {
+                String msg = "Duplicate key: '" + name + "'";
+                return addErrorAndRecover(msg, tokenName, tokenObjectEnd);
+            }
+
+            Token colon;
+            if (!readToken(colon) || colon.type_ != tokenMemberSeparator) {
+                return addErrorAndRecover("Missing ':' after object member name", colon,
+                                          tokenObjectEnd);
+            }
+            Value &value = currentValue()[name];
+            nodes_.push(&value);
+            bool ok = readValue();
+            nodes_.pop();
+            if (!ok) // error already put
+                return recoverFromError(tokenObjectEnd);
+
+            Token comma;
+            if (!readToken(comma) ||
+                (comma.type_ != tokenObjectEnd && comma.type_ != tokenArraySeparator &&
+                 comma.type_ != tokenComment)) {
+                return addErrorAndRecover("Missing ',' or '}' in object declaration",
+                                          comma, tokenObjectEnd);
+            }
+            bool finalizeTokenOk = true;
+            while (comma.type_ == tokenComment && finalizeTokenOk)
+                finalizeTokenOk = readToken(comma);
+            if (comma.type_ == tokenObjectEnd)
+                return true;
+        }
+        return addErrorAndRecover("Missing '}' or object member name", tokenName,
+                                  tokenObjectEnd);
+    }
+
+    bool OurReader::readArray(Token &token) {
+        Value init(arrayValue);
+        currentValue().swapPayload(init);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        int index = 0;
+        for (;;) {
+            skipSpaces();
+            if (current_ != end_ && *current_ == ']' &&
+                (index == 0 ||
+                 (features_.allowTrailingCommas_ &&
+                  !features_.allowDroppedNullPlaceholders_))) // empty array or trailing
+                // comma
+            {
+                Token endArray;
+                readToken(endArray);
+                return true;
+            }
+            Value &value = currentValue()[index++];
+            nodes_.push(&value);
+            bool ok = readValue();
+            nodes_.pop();
+            if (!ok) // error already put
+                return recoverFromError(tokenArrayEnd);
+
+            Token currentToken;
+            // Accept Comment after last item in the array.
+            ok = readToken(currentToken);
+            while (currentToken.type_ == tokenComment && ok) {
+                ok = readToken(currentToken);
+            }
+            bool badTokenType = (currentToken.type_ != tokenArraySeparator &&
+                                 currentToken.type_ != tokenArrayEnd);
+            if (!ok || badTokenType) {
+                return addErrorAndRecover("Missing ',' or ']' in array declaration",
+                                          currentToken, tokenArrayEnd);
+            }
+            if (currentToken.type_ == tokenArrayEnd)
+                break;
+        }
+        return true;
+    }
+
+    bool OurReader::decodeNumber(Token &token) {
+        Value decoded;
+        if (!decodeNumber(token, decoded))
+            return false;
+        currentValue().swapPayload(decoded);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        currentValue().setOffsetLimit(token.end_ - begin_);
+        return true;
+    }
+
+    bool OurReader::decodeNumber(Token &token, Value &decoded) {
+        // Attempts to parse the number as an integer. If the number is
+        // larger than the maximum supported value of an integer then
+        // we decode the number as a double.
+        Location current = token.start_;
+        const bool isNegative = *current == '-';
+        if (isNegative) {
+            ++current;
+        }
+
+        // We assume we can represent the largest and smallest integer types as
+        // unsigned integers with separate sign. This is only true if they can fit
+        // into an unsigned integer.
+        static_assert(Value::maxLargestInt <= Value::maxLargestUInt,
+                      "Int must be smaller than UInt");
+
+        // We need to convert minLargestInt into a positive number. The easiest way
+        // to do this conversion is to assume our "threshold" value of minLargestInt
+        // divided by 10 can fit in maxLargestInt when absolute valued. This should
+        // be a safe assumption.
+        static_assert(Value::minLargestInt <= -Value::maxLargestInt,
+                      "The absolute value of minLargestInt must be greater than or "
+                      "equal to maxLargestInt");
+        static_assert(Value::minLargestInt / 10 >= -Value::maxLargestInt,
+                      "The absolute value of minLargestInt must be only 1 magnitude "
+                      "larger than maxLargest Int");
+
+        static constexpr Value::LargestUInt positive_threshold =
+                Value::maxLargestUInt / 10;
+        static constexpr Value::UInt positive_last_digit = Value::maxLargestUInt % 10;
+
+        // For the negative values, we have to be more careful. Since typically
+        // -Value::minLargestInt will cause an overflow, we first divide by 10 and
+        // then take the inverse. This assumes that minLargestInt is only a single
+        // power of 10 different in magnitude, which we check above. For the last
+        // digit, we take the modulus before negating for the same reason.
+        static constexpr auto negative_threshold =
+                Value::LargestUInt(-(Value::minLargestInt / 10));
+        static constexpr auto negative_last_digit =
+                Value::UInt(-(Value::minLargestInt % 10));
+
+        const Value::LargestUInt threshold =
+                isNegative ? negative_threshold : positive_threshold;
+        const Value::UInt max_last_digit =
+                isNegative ? negative_last_digit : positive_last_digit;
+
+        Value::LargestUInt value = 0;
+        while (current < token.end_) {
+            Char c = *current++;
+            if (c < '0' || c > '9')
+                return decodeDouble(token, decoded);
+
+            const auto digit(static_cast<Value::UInt>(c - '0'));
+            if (value >= threshold) {
+                // We've hit or exceeded the max value divided by 10 (rounded down). If
+                // a) we've only just touched the limit, meaing value == threshold,
+                // b) this is the last digit, or
+                // c) it's small enough to fit in that rounding delta, we're okay.
+                // Otherwise treat this number as a double to avoid overflow.
+                if (value > threshold || current != token.end_ ||
+                    digit > max_last_digit) {
+                    return decodeDouble(token, decoded);
+                }
+            }
+            value = value * 10 + digit;
+        }
+
+        if (isNegative) {
+            // We use the same magnitude assumption here, just in case.
+            const auto last_digit = static_cast<Value::UInt>(value % 10);
+            decoded = -Value::LargestInt(value / 10) * 10 - last_digit;
+        } else if (value <= Value::LargestUInt(Value::maxLargestInt)) {
+            decoded = Value::LargestInt(value);
+        } else {
+            decoded = value;
+        }
+
+        return true;
+    }
+
+    bool OurReader::decodeDouble(Token &token) {
+        Value decoded;
+        if (!decodeDouble(token, decoded))
+            return false;
+        currentValue().swapPayload(decoded);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        currentValue().setOffsetLimit(token.end_ - begin_);
+        return true;
+    }
+
+    bool OurReader::decodeDouble(Token &token, Value &decoded) {
+        double value = 0;
+        const String buffer(token.start_, token.end_);
+        IStringStream is(buffer);
+        if (!(is >> value)) {
+            return addError(
+                    "'" + String(token.start_, token.end_) + "' is not a number.", token);
+        }
+        decoded = value;
+        return true;
+    }
+
+    bool OurReader::decodeString(Token &token) {
+        String decoded_string;
+        if (!decodeString(token, decoded_string))
+            return false;
+        Value decoded(decoded_string);
+        currentValue().swapPayload(decoded);
+        currentValue().setOffsetStart(token.start_ - begin_);
+        currentValue().setOffsetLimit(token.end_ - begin_);
+        return true;
+    }
+
+    bool OurReader::decodeString(Token &token, String &decoded) {
+        decoded.reserve(static_cast<size_t>(token.end_ - token.start_ - 2));
+        Location current = token.start_ + 1; // skip '"'
+        Location end = token.end_ - 1;       // do not include '"'
+        while (current != end) {
+            Char c = *current++;
+            if (c == '"')
+                break;
+            if (c == '\\') {
+                if (current == end)
+                    return addError("Empty escape sequence in string", token, current);
+                Char escape = *current++;
+                switch (escape) {
+                    case '"':
+                        decoded += '"';
+                        break;
+                    case '/':
+                        decoded += '/';
+                        break;
+                    case '\\':
+                        decoded += '\\';
+                        break;
+                    case 'b':
+                        decoded += '\b';
+                        break;
+                    case 'f':
+                        decoded += '\f';
+                        break;
+                    case 'n':
+                        decoded += '\n';
+                        break;
+                    case 'r':
+                        decoded += '\r';
+                        break;
+                    case 't':
+                        decoded += '\t';
+                        break;
+                    case 'u': {
+                        unsigned int unicode;
+                        if (!decodeUnicodeCodePoint(token, current, end, unicode))
+                            return false;
+                        decoded += codePointToUTF8(unicode);
+                    }
+                        break;
+                    default:
+                        return addError("Bad escape sequence in string", token, current);
+                }
+            } else {
+                decoded += c;
+            }
+        }
+        return true;
+    }
+
+    bool OurReader::decodeUnicodeCodePoint(Token &token, Location &current,
+                                           Location end, unsigned int &unicode) {
+
+        if (!decodeUnicodeEscapeSequence(token, current, end, unicode))
+            return false;
+        if (unicode >= 0xD800 && unicode <= 0xDBFF) {
+            // surrogate pairs
+            if (end - current < 6)
+                return addError(
+                        "additional six characters expected to parse unicode surrogate pair.",
+                        token, current);
+            if (*(current++) == '\\' && *(current++) == 'u') {
+                unsigned int surrogatePair;
+                if (decodeUnicodeEscapeSequence(token, current, end, surrogatePair)) {
+                    unicode = 0x10000 + ((unicode & 0x3FF) << 10) + (surrogatePair & 0x3FF);
+                } else
+                    return false;
+            } else
+                return addError("expecting another \\u token to begin the second half of "
+                                "a unicode surrogate pair",
+                                token, current);
+        }
+        return true;
+    }
+
+    bool OurReader::decodeUnicodeEscapeSequence(Token &token, Location &current,
+                                                Location end,
+                                                unsigned int &ret_unicode) {
+        if (end - current < 4)
+            return addError(
+                    "Bad unicode escape sequence in string: four digits expected.", token,
+                    current);
+        int unicode = 0;
+        for (int index = 0; index < 4; ++index) {
+            Char c = *current++;
+            unicode *= 16;
+            if (c >= '0' && c <= '9')
+                unicode += c - '0';
+            else if (c >= 'a' && c <= 'f')
+                unicode += c - 'a' + 10;
+            else if (c >= 'A' && c <= 'F')
+                unicode += c - 'A' + 10;
+            else
+                return addError(
+                        "Bad unicode escape sequence in string: hexadecimal digit expected.",
+                        token, current);
+        }
+        ret_unicode = static_cast<unsigned int>(unicode);
+        return true;
+    }
+
+    bool OurReader::addError(const String &message, Token &token, Location extra) {
+        ErrorInfo info;
+        info.token_ = token;
+        info.message_ = message;
+        info.extra_ = extra;
+        errors_.push_back(info);
+        return false;
+    }
+
+    bool OurReader::recoverFromError(TokenType skipUntilToken) {
+        size_t errorCount = errors_.size();
+        Token skip;
+        for (;;) {
+            if (!readToken(skip))
+                errors_.resize(errorCount); // discard errors caused by recovery
+            if (skip.type_ == skipUntilToken || skip.type_ == tokenEndOfStream)
+                break;
+        }
+        errors_.resize(errorCount);
+        return false;
+    }
+
+    bool OurReader::addErrorAndRecover(const String &message, Token &token,
+                                       TokenType skipUntilToken) {
+        addError(message, token);
+        return recoverFromError(skipUntilToken);
+    }
+
+    Value &OurReader::currentValue() { return *(nodes_.top()); }
+
+    OurReader::Char OurReader::getNextChar() {
+        if (current_ == end_)
+            return 0;
+        return *current_++;
+    }
+
+    void OurReader::getLocationLineAndColumn(Location location, int &line,
+                                             int &column) const {
+        Location current = begin_;
+        Location lastLineStart = current;
+        line = 0;
+        while (current < location && current != end_) {
+            Char c = *current++;
+            if (c == '\r') {
+                if (*current == '\n')
+                    ++current;
+                lastLineStart = current;
+                ++line;
+            } else if (c == '\n') {
+                lastLineStart = current;
+                ++line;
+            }
+        }
+        // column & line start at 1
+        column = int(location - lastLineStart) + 1;
+        ++line;
+    }
+
+    String OurReader::getLocationLineAndColumn(Location location) const {
+        int line, column;
+        getLocationLineAndColumn(location, line, column);
+        char buffer[18 + 16 + 16 + 1];
+        jsoncpp_snprintf(buffer, sizeof(buffer), "Line %d, Column %d", line, column);
+        return buffer;
+    }
+
+    String OurReader::getFormattedErrorMessages() const {
+        String formattedMessage;
+        for (const auto &error : errors_) {
+            formattedMessage +=
+                    "* " + getLocationLineAndColumn(error.token_.start_) + "\n";
+            formattedMessage += "  " + error.message_ + "\n";
+            if (error.extra_)
+                formattedMessage +=
+                        "See " + getLocationLineAndColumn(error.extra_) + " for detail.\n";
+        }
+        return formattedMessage;
+    }
+
+    std::vector<OurReader::StructuredError> OurReader::getStructuredErrors() const {
+        std::vector<OurReader::StructuredError> allErrors;
+        for (const auto &error : errors_) {
+            OurReader::StructuredError structured;
+            structured.offset_start = error.token_.start_ - begin_;
+            structured.offset_limit = error.token_.end_ - begin_;
+            structured.message = error.message_;
+            allErrors.push_back(structured);
+        }
+        return allErrors;
+    }
+
+    class OurCharReader : public CharReader {
+        bool const collectComments_;
+        OurReader reader_;
+
+    public:
+        OurCharReader(bool collectComments, OurFeatures const &features)
+                : collectComments_(collectComments), reader_(features) {}
+
+        bool parse(char const *beginDoc, char const *endDoc, Value *root,
+                   String *errs) override {
+            bool ok = reader_.parse(beginDoc, endDoc, *root, collectComments_);
+            if (errs) {
+                *errs = reader_.getFormattedErrorMessages();
+            }
+            return ok;
+        }
+    };
+
+    CharReaderBuilder::CharReaderBuilder() { setDefaults(&settings_); }
+
+    CharReaderBuilder::~CharReaderBuilder() = default;
+
+    CharReader *CharReaderBuilder::newCharReader() const {
+        bool collectComments = settings_["collectComments"].asBool();
+        OurFeatures features = OurFeatures::all();
+        features.allowComments_ = settings_["allowComments"].asBool();
+        features.allowTrailingCommas_ = settings_["allowTrailingCommas"].asBool();
+        features.strictRoot_ = settings_["strictRoot"].asBool();
+        features.allowDroppedNullPlaceholders_ =
+                settings_["allowDroppedNullPlaceholders"].asBool();
+        features.allowNumericKeys_ = settings_["allowNumericKeys"].asBool();
+        features.allowSingleQuotes_ = settings_["allowSingleQuotes"].asBool();
+
+        // Stack limit is always a size_t, so we get this as an unsigned int
+        // regardless of it we have 64-bit integer support enabled.
+        features.stackLimit_ = static_cast<size_t>(settings_["stackLimit"].asUInt());
+        features.failIfExtra_ = settings_["failIfExtra"].asBool();
+        features.rejectDupKeys_ = settings_["rejectDupKeys"].asBool();
+        features.allowSpecialFloats_ = settings_["allowSpecialFloats"].asBool();
+        features.skipBom_ = settings_["skipBom"].asBool();
+        return new OurCharReader(collectComments, features);
+    }
+
+    bool CharReaderBuilder::validate(Json::Value *invalid) const {
+        static const auto &valid_keys = *new std::set<String>{
+                "collectComments",
+                "allowComments",
+                "allowTrailingCommas",
+                "strictRoot",
+                "allowDroppedNullPlaceholders",
+                "allowNumericKeys",
+                "allowSingleQuotes",
+                "stackLimit",
+                "failIfExtra",
+                "rejectDupKeys",
+                "allowSpecialFloats",
+                "skipBom",
+        };
+        for (auto si = settings_.begin(); si != settings_.end(); ++si) {
+            auto key = si.name();
+            if (valid_keys.count(key))
+                continue;
+            if (invalid)
+                (*invalid)[std::move(key)] = *si;
+            else
+                return false;
+        }
+        return invalid ? invalid->empty() : true;
+    }
+
+    Value &CharReaderBuilder::operator[](const String &key) {
+        return settings_[key];
+    }
+
+// static
+    void CharReaderBuilder::strictMode(Json::Value *settings) {
+        //! [CharReaderBuilderStrictMode]
+        (*settings)["allowComments"] = false;
+        (*settings)["allowTrailingCommas"] = false;
+        (*settings)["strictRoot"] = true;
+        (*settings)["allowDroppedNullPlaceholders"] = false;
+        (*settings)["allowNumericKeys"] = false;
+        (*settings)["allowSingleQuotes"] = false;
+        (*settings)["stackLimit"] = 1000;
+        (*settings)["failIfExtra"] = true;
+        (*settings)["rejectDupKeys"] = true;
+        (*settings)["allowSpecialFloats"] = false;
+        (*settings)["skipBom"] = true;
+        //! [CharReaderBuilderStrictMode]
+    }
+
+// static
+    void CharReaderBuilder::setDefaults(Json::Value *settings) {
+        //! [CharReaderBuilderDefaults]
+        (*settings)["collectComments"] = true;
+        (*settings)["allowComments"] = true;
+        (*settings)["allowTrailingCommas"] = true;
+        (*settings)["strictRoot"] = false;
+        (*settings)["allowDroppedNullPlaceholders"] = false;
+        (*settings)["allowNumericKeys"] = false;
+        (*settings)["allowSingleQuotes"] = false;
+        (*settings)["stackLimit"] = 1000;
+        (*settings)["failIfExtra"] = false;
+        (*settings)["rejectDupKeys"] = false;
+        (*settings)["allowSpecialFloats"] = false;
+        (*settings)["skipBom"] = true;
+        //! [CharReaderBuilderDefaults]
+    }
+
+//////////////////////////////////
+// global functions
+
+    bool parseFromStream(CharReader::Factory const &fact, IStream &sin, Value *root,
+                         String *errs) {
+        OStringStream ssin;
+        ssin << sin.rdbuf();
+        String doc = ssin.str();
+        char const *begin = doc.data();
+        char const *end = begin + doc.size();
+        // Note that we do not actually need a null-terminator.
+        CharReaderPtr const reader(fact.newCharReader());
+        return reader->parse(begin, end, root, errs);
+    }
+
+    IStream &operator>>(IStream &sin, Value &root) {
+        CharReaderBuilder b;
+        String errs;
+        bool ok = parseFromStream(b, sin, &root, &errs);
+        if (!ok) {
+            throwRuntimeError(errs);
+        }
+        return sin;
+    }
+
+} // namespace Json
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: src/lib_json/json_reader.cpp
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: src/lib_json/json_valueiterator.inl
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+// included by json_value.cpp
+
+namespace Json {
+
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// class ValueIteratorBase
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+
+    ValueIteratorBase::ValueIteratorBase() : current_() {}
+
+    ValueIteratorBase::ValueIteratorBase(
+            const Value::ObjectValues::iterator &current)
+            : current_(current), isNull_(false) {}
+
+    Value &ValueIteratorBase::deref() { return current_->second; }
+
+    const Value &ValueIteratorBase::deref() const { return current_->second; }
+
+    void ValueIteratorBase::increment() { ++current_; }
+
+    void ValueIteratorBase::decrement() { --current_; }
+
+    ValueIteratorBase::difference_type
+    ValueIteratorBase::computeDistance(const SelfType &other) const {
+        // Iterator for null value are initialized using the default
+        // constructor, which initialize current_ to the default
+        // std::map::iterator. As begin() and end() are two instance
+        // of the default std::map::iterator, they can not be compared.
+        // To allow this, we handle this comparison specifically.
+        if (isNull_ && other.isNull_) {
+            return 0;
+        }
+
+        // Usage of std::distance is not portable (does not compile with Sun Studio 12
+        // RogueWave STL,
+        // which is the one used by default).
+        // Using a portable hand-made version for non random iterator instead:
+        //   return difference_type( std::distance( current_, other.current_ ) );
+        difference_type myDistance = 0;
+        for (Value::ObjectValues::iterator it = current_; it != other.current_;
+             ++it) {
+            ++myDistance;
+        }
+        return myDistance;
+    }
+
+    bool ValueIteratorBase::isEqual(const SelfType &other) const {
+        if (isNull_) {
+            return other.isNull_;
+        }
+        return current_ == other.current_;
+    }
+
+    void ValueIteratorBase::copy(const SelfType &other) {
+        current_ = other.current_;
+        isNull_ = other.isNull_;
+    }
+
+    Value ValueIteratorBase::key() const {
+        const Value::CZString czstring = (*current_).first;
+        if (czstring.data()) {
+            if (czstring.isStaticString())
+                return Value(StaticString(czstring.data()));
+            return Value(czstring.data(), czstring.data() + czstring.length());
+        }
+        return Value(czstring.index());
+    }
+
+    UInt ValueIteratorBase::index() const {
+        const Value::CZString czstring = (*current_).first;
+        if (!czstring.data())
+            return czstring.index();
+        return Value::UInt(-1);
+    }
+
+    String ValueIteratorBase::name() const {
+        char const *keey;
+        char const *end;
+        keey = memberName(&end);
+        if (!keey)
+            return String();
+        return String(keey, end);
+    }
+
+    char const *ValueIteratorBase::memberName() const {
+        const char *cname = (*current_).first.data();
+        return cname ? cname : "";
+    }
+
+    char const *ValueIteratorBase::memberName(char const **end) const {
+        const char *cname = (*current_).first.data();
+        if (!cname) {
+            *end = nullptr;
+            return nullptr;
+        }
+        *end = cname + (*current_).first.length();
+        return cname;
+    }
+
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// class ValueConstIterator
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+
+    ValueConstIterator::ValueConstIterator() = default;
+
+    ValueConstIterator::ValueConstIterator(
+            const Value::ObjectValues::iterator &current)
+            : ValueIteratorBase(current) {}
+
+    ValueConstIterator::ValueConstIterator(ValueIterator const &other)
+            : ValueIteratorBase(other) {}
+
+    ValueConstIterator &ValueConstIterator::
+    operator=(const ValueIteratorBase &other) {
+        copy(other);
+        return *this;
+    }
+
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// class ValueIterator
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+
+    ValueIterator::ValueIterator() = default;
+
+    ValueIterator::ValueIterator(const Value::ObjectValues::iterator &current)
+            : ValueIteratorBase(current) {}
+
+    ValueIterator::ValueIterator(const ValueConstIterator &other)
+            : ValueIteratorBase(other) {
+        throwRuntimeError("ConstIterator to Iterator should never be allowed.");
+    }
+
+    ValueIterator::ValueIterator(const ValueIterator &other) = default;
+
+    ValueIterator &ValueIterator::operator=(const SelfType &other) {
+        copy(other);
+        return *this;
+    }
+
+} // namespace Json
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: src/lib_json/json_valueiterator.inl
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: src/lib_json/json_value.cpp
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2011 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#if !defined(JSON_IS_AMALGAMATION)
+                                                                                                                        #include <json/assertions.h>
+#include <json/value.h>
+#include <json/writer.h>
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+#include <algorithm>
+#include <cassert>
+#include <cmath>
+#include <cstddef>
+#include <cstring>
+#include <iostream>
+#include <sstream>
+#include <utility>
+
+// Provide implementation equivalent of std::snprintf for older _MSC compilers
+#if defined(_MSC_VER) && _MSC_VER < 1900
+                                                                                                                        #include <stdarg.h>
+static int msvc_pre1900_c99_vsnprintf(char* outBuf, size_t size,
+                                      const char* format, va_list ap) {
+  int count = -1;
+  if (size != 0)
+    count = _vsnprintf_s(outBuf, size, _TRUNCATE, format, ap);
+  if (count == -1)
+    count = _vscprintf(format, ap);
+  return count;
+}
+
+int JSON_API msvc_pre1900_c99_snprintf(char* outBuf, size_t size,
+                                       const char* format, ...) {
+  va_list ap;
+  va_start(ap, format);
+  const int count = msvc_pre1900_c99_vsnprintf(outBuf, size, format, ap);
+  va_end(ap);
+  return count;
+}
+#endif
+
+// Disable warning C4702 : unreachable code
+#if defined(_MSC_VER)
+#pragma warning(disable : 4702)
+#endif
+
+#define JSON_ASSERT_UNREACHABLE assert(false)
+
+namespace Json {
+    template<typename T>
+    static std::unique_ptr<T> cloneUnique(const std::unique_ptr<T> &p) {
+        std::unique_ptr<T> r;
+        if (p) {
+            r = std::unique_ptr<T>(new T(*p));
+        }
+        return r;
+    }
+
+// This is a walkaround to avoid the static initialization of Value::null.
+// kNull must be word-aligned to avoid crashing on ARM.  We use an alignment of
+// 8 (instead of 4) as a bit of future-proofing.
+#if defined(__ARMEL__)
+#define ALIGNAS(byte_alignment) __attribute__((aligned(byte_alignment)))
+#else
+#define ALIGNAS(byte_alignment)
+#endif
+
+// static
+    Value const &Value::nullSingleton() {
+        static Value const nullStatic;
+        return nullStatic;
+    }
+
+#if JSON_USE_NULLREF
+// for backwards compatibility, we'll leave these global references around, but
+// DO NOT use them in JSONCPP library code any more!
+// static
+    Value const &Value::null = Value::nullSingleton();
+
+// static
+    Value const &Value::nullRef = Value::nullSingleton();
+#endif
+
+#if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+                                                                                                                            template <typename T, typename U>
+static inline bool InRange(double d, T min, U max) {
+  // The casts can lose precision, but we are looking only for
+  // an approximate range. Might fail on edge cases though. ~cdunn
+  return d >= static_cast<double>(min) && d <= static_cast<double>(max);
+}
+#else  // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+
+    static inline double integerToDouble(Json::UInt64 value) {
+        return static_cast<double>(Int64(value / 2)) * 2.0 +
+               static_cast<double>(Int64(value & 1));
+    }
+
+    template<typename T>
+    static inline double integerToDouble(T value) {
+        return static_cast<double>(value);
+    }
+
+    template<typename T, typename U>
+    static inline bool InRange(double d, T min, U max) {
+        return d >= integerToDouble(min) && d <= integerToDouble(max);
+    }
+
+#endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+
+/** Duplicates the specified string value.
+ * @param value Pointer to the string to duplicate. Must be zero-terminated if
+ *              length is "unknown".
+ * @param length Length of the value. if equals to unknown, then it will be
+ *               computed using strlen(value).
+ * @return Pointer on the duplicate instance of string.
+ */
+    static inline char *duplicateStringValue(const char *value, size_t length) {
+        // Avoid an integer overflow in the call to malloc below by limiting length
+        // to a sane value.
+        if (length >= static_cast<size_t>(Value::maxInt))
+            length = Value::maxInt - 1;
+
+        auto newString = static_cast<char *>(malloc(length + 1));
+        if (newString == nullptr) {
+            throwRuntimeError("in Json::Value::duplicateStringValue(): "
+                              "Failed to allocate string value buffer");
+        }
+        memcpy(newString, value, length);
+        newString[length] = 0;
+        return newString;
+    }
+
+/* Record the length as a prefix.
+ */
+    static inline char *duplicateAndPrefixStringValue(const char *value,
+                                                      unsigned int length) {
+        // Avoid an integer overflow in the call to malloc below by limiting length
+        // to a sane value.
+        JSON_ASSERT_MESSAGE(length <= static_cast<unsigned>(Value::maxInt) -
+                                      sizeof(unsigned) - 1U,
+                            "in Json::Value::duplicateAndPrefixStringValue(): "
+                            "length too big for prefixing");
+        size_t actualLength = sizeof(length) + length + 1;
+        auto newString = static_cast<char *>(malloc(actualLength));
+        if (newString == nullptr) {
+            throwRuntimeError("in Json::Value::duplicateAndPrefixStringValue(): "
+                              "Failed to allocate string value buffer");
+        }
+        *reinterpret_cast<unsigned *>(newString) = length;
+        memcpy(newString + sizeof(unsigned), value, length);
+        newString[actualLength - 1U] =
+                0; // to avoid buffer over-run accidents by users later
+        return newString;
+    }
+
+    inline static void decodePrefixedString(bool isPrefixed, char const *prefixed,
+                                            unsigned *length, char const **value) {
+        if (!isPrefixed) {
+            *length = static_cast<unsigned>(strlen(prefixed));
+            *value = prefixed;
+        } else {
+            *length = *reinterpret_cast<unsigned const *>(prefixed);
+            *value = prefixed + sizeof(unsigned);
+        }
+    }
+/** Free the string duplicated by
+ * duplicateStringValue()/duplicateAndPrefixStringValue().
+ */
+#if JSONCPP_USING_SECURE_MEMORY
+                                                                                                                            static inline void releasePrefixedStringValue(char* value) {
+  unsigned length = 0;
+  char const* valueDecoded;
+  decodePrefixedString(true, value, &length, &valueDecoded);
+  size_t const size = sizeof(unsigned) + length + 1U;
+  memset(value, 0, size);
+  free(value);
+}
+static inline void releaseStringValue(char* value, unsigned length) {
+  // length==0 => we allocated the strings memory
+  size_t size = (length == 0) ? strlen(value) : length;
+  memset(value, 0, size);
+  free(value);
+}
+#else  // !JSONCPP_USING_SECURE_MEMORY
+
+    static inline void releasePrefixedStringValue(char *value) { free(value); }
+
+    static inline void releaseStringValue(char *value, unsigned) { free(value); }
+
+#endif // JSONCPP_USING_SECURE_MEMORY
+
+} // namespace Json
+
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// ValueInternals...
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+#if !defined(JSON_IS_AMALGAMATION)
+
+#include "json_valueiterator.inl"
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+namespace Json {
+
+#if JSON_USE_EXCEPTION
+
+    Exception::Exception(String msg) : msg_(std::move(msg)) {}
+
+    Exception::~Exception() noexcept = default;
+
+    char const *Exception::what() const noexcept { return msg_.c_str(); }
+
+    RuntimeError::RuntimeError(String const &msg) : Exception(msg) {}
+
+    LogicError::LogicError(String const &msg) : Exception(msg) {}
+
+    JSONCPP_NORETURN void throwRuntimeError(String const &msg) {
+        throw RuntimeError(msg);
+    }
+
+    JSONCPP_NORETURN void throwLogicError(String const &msg) {
+        throw LogicError(msg);
+    }
+
+#else // !JSON_USE_EXCEPTION
+                                                                                                                            JSONCPP_NORETURN void throwRuntimeError(String const& msg) {
+  std::cerr << msg << std::endl;
+  abort();
+}
+JSONCPP_NORETURN void throwLogicError(String const& msg) {
+  std::cerr << msg << std::endl;
+  abort();
+}
+#endif
+
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// class Value::CZString
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+
+// Notes: policy_ indicates if the string was allocated when
+// a string is stored.
+
+    Value::CZString::CZString(ArrayIndex index) : cstr_(nullptr), index_(index) {}
+
+    Value::CZString::CZString(char const *str, unsigned length,
+                              DuplicationPolicy allocate)
+            : cstr_(str) {
+        // allocate != duplicate
+        storage_.policy_ = allocate & 0x3;
+        storage_.length_ = length & 0x3FFFFFFF;
+    }
+
+    Value::CZString::CZString(const CZString &other) {
+        cstr_ = (other.storage_.policy_ != noDuplication && other.cstr_ != nullptr
+                 ? duplicateStringValue(other.cstr_, other.storage_.length_)
+                 : other.cstr_);
+        storage_.policy_ =
+                static_cast<unsigned>(
+                        other.cstr_
+                        ? (static_cast<DuplicationPolicy>(other.storage_.policy_) ==
+                           noDuplication
+                           ? noDuplication
+                           : duplicate)
+                        : static_cast<DuplicationPolicy>(other.storage_.policy_)) &
+                3U;
+        storage_.length_ = other.storage_.length_;
+    }
+
+    Value::CZString::CZString(CZString &&other)
+            : cstr_(other.cstr_), index_(other.index_) {
+        other.cstr_ = nullptr;
+    }
+
+    Value::CZString::~CZString() {
+        if (cstr_ && storage_.policy_ == duplicate) {
+            releaseStringValue(const_cast<char *>(cstr_),
+                               storage_.length_ + 1U); // +1 for null terminating
+            // character for sake of
+            // completeness but not actually
+            // necessary
+        }
+    }
+
+    void Value::CZString::swap(CZString &other) {
+        std::swap(cstr_, other.cstr_);
+        std::swap(index_, other.index_);
+    }
+
+    Value::CZString &Value::CZString::operator=(const CZString &other) {
+        cstr_ = other.cstr_;
+        index_ = other.index_;
+        return *this;
+    }
+
+    Value::CZString &Value::CZString::operator=(CZString &&other) {
+        cstr_ = other.cstr_;
+        index_ = other.index_;
+        other.cstr_ = nullptr;
+        return *this;
+    }
+
+    bool Value::CZString::operator<(const CZString &other) const {
+        if (!cstr_)
+            return index_ < other.index_;
+        // return strcmp(cstr_, other.cstr_) < 0;
+        // Assume both are strings.
+        unsigned this_len = this->storage_.length_;
+        unsigned other_len = other.storage_.length_;
+        unsigned min_len = std::min<unsigned>(this_len, other_len);
+        JSON_ASSERT(this->cstr_ && other.cstr_);
+        int comp = memcmp(this->cstr_, other.cstr_, min_len);
+        if (comp < 0)
+            return true;
+        if (comp > 0)
+            return false;
+        return (this_len < other_len);
+    }
+
+    bool Value::CZString::operator==(const CZString &other) const {
+        if (!cstr_)
+            return index_ == other.index_;
+        // return strcmp(cstr_, other.cstr_) == 0;
+        // Assume both are strings.
+        unsigned this_len = this->storage_.length_;
+        unsigned other_len = other.storage_.length_;
+        if (this_len != other_len)
+            return false;
+        JSON_ASSERT(this->cstr_ && other.cstr_);
+        int comp = memcmp(this->cstr_, other.cstr_, this_len);
+        return comp == 0;
+    }
+
+    ArrayIndex Value::CZString::index() const { return index_; }
+
+// const char* Value::CZString::c_str() const { return cstr_; }
+    const char *Value::CZString::data() const { return cstr_; }
+
+    unsigned Value::CZString::length() const { return storage_.length_; }
+
+    bool Value::CZString::isStaticString() const {
+        return storage_.policy_ == noDuplication;
+    }
+
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// class Value::Value
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+// //////////////////////////////////////////////////////////////////
+
+/*! \internal Default constructor initialization must be equivalent to:
+ * memset( this, 0, sizeof(Value) )
+ * This optimization is used in ValueInternalMap fast allocator.
+ */
+    Value::Value(ValueType type) {
+        static char const emptyString[] = "";
+        initBasic(type);
+        switch (type) {
+            case nullValue:
+                break;
+            case intValue:
+            case uintValue:
+                value_.int_ = 0;
+                break;
+            case realValue:
+                value_.real_ = 0.0;
+                break;
+            case stringValue:
+                // allocated_ == false, so this is safe.
+                value_.string_ = const_cast<char *>(static_cast<char const *>(emptyString));
+                break;
+            case arrayValue:
+            case objectValue:
+                value_.map_ = new ObjectValues();
+                break;
+            case booleanValue:
+                value_.bool_ = false;
+                break;
+            default:
+                JSON_ASSERT_UNREACHABLE;
+        }
+    }
+
+    Value::Value(Int value) {
+        initBasic(intValue);
+        value_.int_ = value;
+    }
+
+    Value::Value(UInt value) {
+        initBasic(uintValue);
+        value_.uint_ = value;
+    }
+
+#if defined(JSON_HAS_INT64)
+
+    Value::Value(Int64 value) {
+        initBasic(intValue);
+        value_.int_ = value;
+    }
+
+    Value::Value(UInt64 value) {
+        initBasic(uintValue);
+        value_.uint_ = value;
+    }
+
+#endif // defined(JSON_HAS_INT64)
+
+    Value::Value(double value) {
+        initBasic(realValue);
+        value_.real_ = value;
+    }
+
+    Value::Value(const char *value) {
+        initBasic(stringValue, true);
+        JSON_ASSERT_MESSAGE(value != nullptr,
+                            "Null Value Passed to Value Constructor");
+        value_.string_ = duplicateAndPrefixStringValue(
+                value, static_cast<unsigned>(strlen(value)));
+    }
+
+    Value::Value(const char *begin, const char *end) {
+        initBasic(stringValue, true);
+        value_.string_ =
+                duplicateAndPrefixStringValue(begin, static_cast<unsigned>(end - begin));
+    }
+
+    Value::Value(const String &value) {
+        initBasic(stringValue, true);
+        value_.string_ = duplicateAndPrefixStringValue(
+                value.data(), static_cast<unsigned>(value.length()));
+    }
+
+    Value::Value(const StaticString &value) {
+        initBasic(stringValue);
+        value_.string_ = const_cast<char *>(value.c_str());
+    }
+
+    Value::Value(bool value) {
+        initBasic(booleanValue);
+        value_.bool_ = value;
+    }
+
+    Value::Value(const Value &other) {
+        dupPayload(other);
+        dupMeta(other);
+    }
+
+    Value::Value(Value &&other) {
+        initBasic(nullValue);
+        swap(other);
+    }
+
+    Value::~Value() {
+        releasePayload();
+        value_.uint_ = 0;
+    }
+
+    Value &Value::operator=(const Value &other) {
+        Value(other).swap(*this);
+        return *this;
+    }
+
+    Value &Value::operator=(Value &&other) {
+        other.swap(*this);
+        return *this;
+    }
+
+    void Value::swapPayload(Value &other) {
+        std::swap(bits_, other.bits_);
+        std::swap(value_, other.value_);
+    }
+
+    void Value::copyPayload(const Value &other) {
+        releasePayload();
+        dupPayload(other);
+    }
+
+    void Value::swap(Value &other) {
+        swapPayload(other);
+        std::swap(comments_, other.comments_);
+        std::swap(start_, other.start_);
+        std::swap(limit_, other.limit_);
+    }
+
+    void Value::copy(const Value &other) {
+        copyPayload(other);
+        dupMeta(other);
+    }
+
+    ValueType Value::type() const {
+        return static_cast<ValueType>(bits_.value_type_);
+    }
+
+    int Value::compare(const Value &other) const {
+        if (*this < other)
+            return -1;
+        if (*this > other)
+            return 1;
+        return 0;
+    }
+
+    bool Value::operator<(const Value &other) const {
+        int typeDelta = type() - other.type();
+        if (typeDelta)
+            return typeDelta < 0;
+        switch (type()) {
+            case nullValue:
+                return false;
+            case intValue:
+                return value_.int_ < other.value_.int_;
+            case uintValue:
+                return value_.uint_ < other.value_.uint_;
+            case realValue:
+                return value_.real_ < other.value_.real_;
+            case booleanValue:
+                return value_.bool_ < other.value_.bool_;
+            case stringValue: {
+                if ((value_.string_ == nullptr) || (other.value_.string_ == nullptr)) {
+                    return other.value_.string_ != nullptr;
+                }
+                unsigned this_len;
+                unsigned other_len;
+                char const *this_str;
+                char const *other_str;
+                decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
+                                     &this_str);
+                decodePrefixedString(other.isAllocated(), other.value_.string_, &other_len,
+                                     &other_str);
+                unsigned min_len = std::min<unsigned>(this_len, other_len);
+                JSON_ASSERT(this_str && other_str);
+                int comp = memcmp(this_str, other_str, min_len);
+                if (comp < 0)
+                    return true;
+                if (comp > 0)
+                    return false;
+                return (this_len < other_len);
+            }
+            case arrayValue:
+            case objectValue: {
+                auto thisSize = value_.map_->size();
+                auto otherSize = other.value_.map_->size();
+                if (thisSize != otherSize)
+                    return thisSize < otherSize;
+                return (*value_.map_) < (*other.value_.map_);
+            }
+            default:
+                JSON_ASSERT_UNREACHABLE;
+        }
+        return false; // unreachable
+    }
+
+    bool Value::operator<=(const Value &other) const { return !(other < *this); }
+
+    bool Value::operator>=(const Value &other) const { return !(*this < other); }
+
+    bool Value::operator>(const Value &other) const { return other < *this; }
+
+    bool Value::operator==(const Value &other) const {
+        if (type() != other.type())
+            return false;
+        switch (type()) {
+            case nullValue:
+                return true;
+            case intValue:
+                return value_.int_ == other.value_.int_;
+            case uintValue:
+                return value_.uint_ == other.value_.uint_;
+            case realValue:
+                return value_.real_ == other.value_.real_;
+            case booleanValue:
+                return value_.bool_ == other.value_.bool_;
+            case stringValue: {
+                if ((value_.string_ == nullptr) || (other.value_.string_ == nullptr)) {
+                    return (value_.string_ == other.value_.string_);
+                }
+                unsigned this_len;
+                unsigned other_len;
+                char const *this_str;
+                char const *other_str;
+                decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
+                                     &this_str);
+                decodePrefixedString(other.isAllocated(), other.value_.string_, &other_len,
+                                     &other_str);
+                if (this_len != other_len)
+                    return false;
+                JSON_ASSERT(this_str && other_str);
+                int comp = memcmp(this_str, other_str, this_len);
+                return comp == 0;
+            }
+            case arrayValue:
+            case objectValue:
+                return value_.map_->size() == other.value_.map_->size() &&
+                       (*value_.map_) == (*other.value_.map_);
+            default:
+                JSON_ASSERT_UNREACHABLE;
+        }
+        return false; // unreachable
+    }
+
+    bool Value::operator!=(const Value &other) const { return !(*this == other); }
+
+    const char *Value::asCString() const {
+        JSON_ASSERT_MESSAGE(type() == stringValue,
+                            "in Json::Value::asCString(): requires stringValue");
+        if (value_.string_ == nullptr)
+            return nullptr;
+        unsigned this_len;
+        char const *this_str;
+        decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
+                             &this_str);
+        return this_str;
+    }
+
+#if JSONCPP_USING_SECURE_MEMORY
+                                                                                                                            unsigned Value::getCStringLength() const {
+  JSON_ASSERT_MESSAGE(type() == stringValue,
+                      "in Json::Value::asCString(): requires stringValue");
+  if (value_.string_ == 0)
+    return 0;
+  unsigned this_len;
+  char const* this_str;
+  decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
+                       &this_str);
+  return this_len;
+}
+#endif
+
+    bool Value::getString(char const **begin, char const **end) const {
+        if (type() != stringValue)
+            return false;
+        if (value_.string_ == nullptr)
+            return false;
+        unsigned length;
+        decodePrefixedString(this->isAllocated(), this->value_.string_, &length,
+                             begin);
+        *end = *begin + length;
+        return true;
+    }
+
+    String Value::asString() const {
+        switch (type()) {
+            case nullValue:
+                return "";
+            case stringValue: {
+                if (value_.string_ == nullptr)
+                    return "";
+                unsigned this_len;
+                char const *this_str;
+                decodePrefixedString(this->isAllocated(), this->value_.string_, &this_len,
+                                     &this_str);
+                return String(this_str, this_len);
+            }
+            case booleanValue:
+                return value_.bool_ ? "true" : "false";
+            case intValue:
+                return valueToString(value_.int_);
+            case uintValue:
+                return valueToString(value_.uint_);
+            case realValue:
+                return valueToString(value_.real_);
+            default:
+                JSON_FAIL_MESSAGE("Type is not convertible to string");
+        }
+    }
+
+    Value::Int Value::asInt() const {
+        switch (type()) {
+            case intValue:
+                JSON_ASSERT_MESSAGE(isInt(), "LargestInt out of Int range");
+                return Int(value_.int_);
+            case uintValue:
+                JSON_ASSERT_MESSAGE(isInt(), "LargestUInt out of Int range");
+                return Int(value_.uint_);
+            case realValue:
+                JSON_ASSERT_MESSAGE(InRange(value_.real_, minInt, maxInt),
+                                    "double out of Int range");
+                return Int(value_.real_);
+            case nullValue:
+                return 0;
+            case booleanValue:
+                return value_.bool_ ? 1 : 0;
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to Int.");
+    }
+
+    Value::UInt Value::asUInt() const {
+        switch (type()) {
+            case intValue:
+                JSON_ASSERT_MESSAGE(isUInt(), "LargestInt out of UInt range");
+                return UInt(value_.int_);
+            case uintValue:
+                JSON_ASSERT_MESSAGE(isUInt(), "LargestUInt out of UInt range");
+                return UInt(value_.uint_);
+            case realValue:
+                JSON_ASSERT_MESSAGE(InRange(value_.real_, 0, maxUInt),
+                                    "double out of UInt range");
+                return UInt(value_.real_);
+            case nullValue:
+                return 0;
+            case booleanValue:
+                return value_.bool_ ? 1 : 0;
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to UInt.");
+    }
+
+#if defined(JSON_HAS_INT64)
+
+    Value::Int64 Value::asInt64() const {
+        switch (type()) {
+            case intValue:
+                return Int64(value_.int_);
+            case uintValue:
+                JSON_ASSERT_MESSAGE(isInt64(), "LargestUInt out of Int64 range");
+                return Int64(value_.uint_);
+            case realValue:
+                JSON_ASSERT_MESSAGE(InRange(value_.real_, minInt64, maxInt64),
+                                    "double out of Int64 range");
+                return Int64(value_.real_);
+            case nullValue:
+                return 0;
+            case booleanValue:
+                return value_.bool_ ? 1 : 0;
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to Int64.");
+    }
+
+    Value::UInt64 Value::asUInt64() const {
+        switch (type()) {
+            case intValue:
+                JSON_ASSERT_MESSAGE(isUInt64(), "LargestInt out of UInt64 range");
+                return UInt64(value_.int_);
+            case uintValue:
+                return UInt64(value_.uint_);
+            case realValue:
+                JSON_ASSERT_MESSAGE(InRange(value_.real_, 0, maxUInt64),
+                                    "double out of UInt64 range");
+                return UInt64(value_.real_);
+            case nullValue:
+                return 0;
+            case booleanValue:
+                return value_.bool_ ? 1 : 0;
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to UInt64.");
+    }
+
+#endif // if defined(JSON_HAS_INT64)
+
+    LargestInt Value::asLargestInt() const {
+#if defined(JSON_NO_INT64)
+        return asInt();
+#else
+        return asInt64();
+#endif
+    }
+
+    LargestUInt Value::asLargestUInt() const {
+#if defined(JSON_NO_INT64)
+        return asUInt();
+#else
+        return asUInt64();
+#endif
+    }
+
+    double Value::asDouble() const {
+        switch (type()) {
+            case intValue:
+                return static_cast<double>(value_.int_);
+            case uintValue:
+#if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+                return static_cast<double>(value_.uint_);
+#else  // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+                return integerToDouble(value_.uint_);
+#endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+            case realValue:
+                return value_.real_;
+            case nullValue:
+                return 0.0;
+            case booleanValue:
+                return value_.bool_ ? 1.0 : 0.0;
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to double.");
+    }
+
+    float Value::asFloat() const {
+        switch (type()) {
+            case intValue:
+                return static_cast<float>(value_.int_);
+            case uintValue:
+#if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+                return static_cast<float>(value_.uint_);
+#else  // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+                // This can fail (silently?) if the value is bigger than MAX_FLOAT.
+                return static_cast<float>(integerToDouble(value_.uint_));
+#endif // if !defined(JSON_USE_INT64_DOUBLE_CONVERSION)
+            case realValue:
+                return static_cast<float>(value_.real_);
+            case nullValue:
+                return 0.0;
+            case booleanValue:
+                return value_.bool_ ? 1.0F : 0.0F;
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to float.");
+    }
+
+    bool Value::asBool() const {
+        switch (type()) {
+            case booleanValue:
+                return value_.bool_;
+            case nullValue:
+                return false;
+            case intValue:
+                return value_.int_ != 0;
+            case uintValue:
+                return value_.uint_ != 0;
+            case realValue: {
+                // According to JavaScript language zero or NaN is regarded as false
+                const auto value_classification = std::fpclassify(value_.real_);
+                return value_classification != FP_ZERO && value_classification != FP_NAN;
+            }
+            default:
+                break;
+        }
+        JSON_FAIL_MESSAGE("Value is not convertible to bool.");
+    }
+
+    bool Value::isConvertibleTo(ValueType other) const {
+        switch (other) {
+            case nullValue:
+                return (isNumeric() && asDouble() == 0.0) ||
+                       (type() == booleanValue && !value_.bool_) ||
+                       (type() == stringValue && asString().empty()) ||
+                       (type() == arrayValue && value_.map_->empty()) ||
+                       (type() == objectValue && value_.map_->empty()) ||
+                       type() == nullValue;
+            case intValue:
+                return isInt() ||
+                       (type() == realValue && InRange(value_.real_, minInt, maxInt)) ||
+                       type() == booleanValue || type() == nullValue;
+            case uintValue:
+                return isUInt() ||
+                       (type() == realValue && InRange(value_.real_, 0, maxUInt)) ||
+                       type() == booleanValue || type() == nullValue;
+            case realValue:
+                return isNumeric() || type() == booleanValue || type() == nullValue;
+            case booleanValue:
+                return isNumeric() || type() == booleanValue || type() == nullValue;
+            case stringValue:
+                return isNumeric() || type() == booleanValue || type() == stringValue ||
+                       type() == nullValue;
+            case arrayValue:
+                return type() == arrayValue || type() == nullValue;
+            case objectValue:
+                return type() == objectValue || type() == nullValue;
+        }
+        JSON_ASSERT_UNREACHABLE;
+        return false;
+    }
+
+/// Number of values in array or object
+    ArrayIndex Value::size() const {
+        switch (type()) {
+            case nullValue:
+            case intValue:
+            case uintValue:
+            case realValue:
+            case booleanValue:
+            case stringValue:
+                return 0;
+            case arrayValue: // size of the array is highest index + 1
+                if (!value_.map_->empty()) {
+                    ObjectValues::const_iterator itLast = value_.map_->end();
+                    --itLast;
+                    return (*itLast).first.index() + 1;
+                }
+                return 0;
+            case objectValue:
+                return ArrayIndex(value_.map_->size());
+        }
+        JSON_ASSERT_UNREACHABLE;
+        return 0; // unreachable;
+    }
+
+    bool Value::empty() const {
+        if (isNull() || isArray() || isObject())
+            return size() == 0U;
+        return false;
+    }
+
+    Value::operator bool() const { return !isNull(); }
+
+    void Value::clear() {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue ||
+                            type() == objectValue,
+                            "in Json::Value::clear(): requires complex value");
+        start_ = 0;
+        limit_ = 0;
+        switch (type()) {
+            case arrayValue:
+            case objectValue:
+                value_.map_->clear();
+                break;
+            default:
+                break;
+        }
+    }
+
+    void Value::resize(ArrayIndex newSize) {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue,
+                            "in Json::Value::resize(): requires arrayValue");
+        if (type() == nullValue)
+            *this = Value(arrayValue);
+        ArrayIndex oldSize = size();
+        if (newSize == 0)
+            clear();
+        else if (newSize > oldSize)
+            this->operator[](newSize - 1);
+        else {
+            for (ArrayIndex index = newSize; index < oldSize; ++index) {
+                value_.map_->erase(index);
+            }
+            JSON_ASSERT(size() == newSize);
+        }
+    }
+
+    Value &Value::operator[](ArrayIndex index) {
+        JSON_ASSERT_MESSAGE(
+                type() == nullValue || type() == arrayValue,
+                "in Json::Value::operator[](ArrayIndex): requires arrayValue");
+        if (type() == nullValue)
+            *this = Value(arrayValue);
+        CZString key(index);
+        auto it = value_.map_->lower_bound(key);
+        if (it != value_.map_->end() && (*it).first == key)
+            return (*it).second;
+
+        ObjectValues::value_type defaultValue(key, nullSingleton());
+        it = value_.map_->insert(it, defaultValue);
+        return (*it).second;
+    }
+
+    Value &Value::operator[](int index) {
+        JSON_ASSERT_MESSAGE(
+                index >= 0,
+                "in Json::Value::operator[](int index): index cannot be negative");
+        return (*this)[ArrayIndex(index)];
+    }
+
+    const Value &Value::operator[](ArrayIndex index) const {
+        JSON_ASSERT_MESSAGE(
+                type() == nullValue || type() == arrayValue,
+                "in Json::Value::operator[](ArrayIndex)const: requires arrayValue");
+        if (type() == nullValue)
+            return nullSingleton();
+        CZString key(index);
+        ObjectValues::const_iterator it = value_.map_->find(key);
+        if (it == value_.map_->end())
+            return nullSingleton();
+        return (*it).second;
+    }
+
+    const Value &Value::operator[](int index) const {
+        JSON_ASSERT_MESSAGE(
+                index >= 0,
+                "in Json::Value::operator[](int index) const: index cannot be negative");
+        return (*this)[ArrayIndex(index)];
+    }
+
+    void Value::initBasic(ValueType type, bool allocated) {
+        setType(type);
+        setIsAllocated(allocated);
+        comments_ = Comments{};
+        start_ = 0;
+        limit_ = 0;
+    }
+
+    void Value::dupPayload(const Value &other) {
+        setType(other.type());
+        setIsAllocated(false);
+        switch (type()) {
+            case nullValue:
+            case intValue:
+            case uintValue:
+            case realValue:
+            case booleanValue:
+                value_ = other.value_;
+                break;
+            case stringValue:
+                if (other.value_.string_ && other.isAllocated()) {
+                    unsigned len;
+                    char const *str;
+                    decodePrefixedString(other.isAllocated(), other.value_.string_, &len,
+                                         &str);
+                    value_.string_ = duplicateAndPrefixStringValue(str, len);
+                    setIsAllocated(true);
+                } else {
+                    value_.string_ = other.value_.string_;
+                }
+                break;
+            case arrayValue:
+            case objectValue:
+                value_.map_ = new ObjectValues(*other.value_.map_);
+                break;
+            default:
+                JSON_ASSERT_UNREACHABLE;
+        }
+    }
+
+    void Value::releasePayload() {
+        switch (type()) {
+            case nullValue:
+            case intValue:
+            case uintValue:
+            case realValue:
+            case booleanValue:
+                break;
+            case stringValue:
+                if (isAllocated())
+                    releasePrefixedStringValue(value_.string_);
+                break;
+            case arrayValue:
+            case objectValue:
+                delete value_.map_;
+                break;
+            default:
+                JSON_ASSERT_UNREACHABLE;
+        }
+    }
+
+    void Value::dupMeta(const Value &other) {
+        comments_ = other.comments_;
+        start_ = other.start_;
+        limit_ = other.limit_;
+    }
+
+// Access an object value by name, create a null member if it does not exist.
+// @pre Type of '*this' is object or null.
+// @param key is null-terminated.
+    Value &Value::resolveReference(const char *key) {
+        JSON_ASSERT_MESSAGE(
+                type() == nullValue || type() == objectValue,
+                "in Json::Value::resolveReference(): requires objectValue");
+        if (type() == nullValue)
+            *this = Value(objectValue);
+        CZString actualKey(key, static_cast<unsigned>(strlen(key)),
+                           CZString::noDuplication); // NOTE!
+        auto it = value_.map_->lower_bound(actualKey);
+        if (it != value_.map_->end() && (*it).first == actualKey)
+            return (*it).second;
+
+        ObjectValues::value_type defaultValue(actualKey, nullSingleton());
+        it = value_.map_->insert(it, defaultValue);
+        Value &value = (*it).second;
+        return value;
+    }
+
+// @param key is not null-terminated.
+    Value &Value::resolveReference(char const *key, char const *end) {
+        JSON_ASSERT_MESSAGE(
+                type() == nullValue || type() == objectValue,
+                "in Json::Value::resolveReference(key, end): requires objectValue");
+        if (type() == nullValue)
+            *this = Value(objectValue);
+        CZString actualKey(key, static_cast<unsigned>(end - key),
+                           CZString::duplicateOnCopy);
+        auto it = value_.map_->lower_bound(actualKey);
+        if (it != value_.map_->end() && (*it).first == actualKey)
+            return (*it).second;
+
+        ObjectValues::value_type defaultValue(actualKey, nullSingleton());
+        it = value_.map_->insert(it, defaultValue);
+        Value &value = (*it).second;
+        return value;
+    }
+
+    Value Value::get(ArrayIndex index, const Value &defaultValue) const {
+        const Value *value = &((*this)[index]);
+        return value == &nullSingleton() ? defaultValue : *value;
+    }
+
+    bool Value::isValidIndex(ArrayIndex index) const { return index < size(); }
+
+    Value const *Value::find(char const *begin, char const *end) const {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == objectValue,
+                            "in Json::Value::find(begin, end): requires "
+                            "objectValue or nullValue");
+        if (type() == nullValue)
+            return nullptr;
+        CZString actualKey(begin, static_cast<unsigned>(end - begin),
+                           CZString::noDuplication);
+        ObjectValues::const_iterator it = value_.map_->find(actualKey);
+        if (it == value_.map_->end())
+            return nullptr;
+        return &(*it).second;
+    }
+
+    Value *Value::demand(char const *begin, char const *end) {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == objectValue,
+                            "in Json::Value::demand(begin, end): requires "
+                            "objectValue or nullValue");
+        return &resolveReference(begin, end);
+    }
+
+    const Value &Value::operator[](const char *key) const {
+        Value const *found = find(key, key + strlen(key));
+        if (!found)
+            return nullSingleton();
+        return *found;
+    }
+
+    Value const &Value::operator[](const String &key) const {
+        Value const *found = find(key.data(), key.data() + key.length());
+        if (!found)
+            return nullSingleton();
+        return *found;
+    }
+
+    Value &Value::operator[](const char *key) {
+        return resolveReference(key, key + strlen(key));
+    }
+
+    Value &Value::operator[](const String &key) {
+        return resolveReference(key.data(), key.data() + key.length());
+    }
+
+    Value &Value::operator[](const StaticString &key) {
+        return resolveReference(key.c_str());
+    }
+
+    Value &Value::append(const Value &value) { return append(Value(value)); }
+
+    Value &Value::append(Value &&value) {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue,
+                            "in Json::Value::append: requires arrayValue");
+        if (type() == nullValue) {
+            *this = Value(arrayValue);
+        }
+        return this->value_.map_->emplace(size(), std::move(value)).first->second;
+    }
+
+    bool Value::insert(ArrayIndex index, const Value &newValue) {
+        return insert(index, Value(newValue));
+    }
+
+    bool Value::insert(ArrayIndex index, Value &&newValue) {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == arrayValue,
+                            "in Json::Value::insert: requires arrayValue");
+        ArrayIndex length = size();
+        if (index > length) {
+            return false;
+        }
+        for (ArrayIndex i = length; i > index; i--) {
+            (*this)[i] = std::move((*this)[i - 1]);
+        }
+        (*this)[index] = std::move(newValue);
+        return true;
+    }
+
+    Value Value::get(char const *begin, char const *end,
+                     Value const &defaultValue) const {
+        Value const *found = find(begin, end);
+        return !found ? defaultValue : *found;
+    }
+
+    Value Value::get(char const *key, Value const &defaultValue) const {
+        return get(key, key + strlen(key), defaultValue);
+    }
+
+    Value Value::get(String const &key, Value const &defaultValue) const {
+        return get(key.data(), key.data() + key.length(), defaultValue);
+    }
+
+    bool Value::removeMember(const char *begin, const char *end, Value *removed) {
+        if (type() != objectValue) {
+            return false;
+        }
+        CZString actualKey(begin, static_cast<unsigned>(end - begin),
+                           CZString::noDuplication);
+        auto it = value_.map_->find(actualKey);
+        if (it == value_.map_->end())
+            return false;
+        if (removed)
+            *removed = std::move(it->second);
+        value_.map_->erase(it);
+        return true;
+    }
+
+    bool Value::removeMember(const char *key, Value *removed) {
+        return removeMember(key, key + strlen(key), removed);
+    }
+
+    bool Value::removeMember(String const &key, Value *removed) {
+        return removeMember(key.data(), key.data() + key.length(), removed);
+    }
+
+    void Value::removeMember(const char *key) {
+        JSON_ASSERT_MESSAGE(type() == nullValue || type() == objectValue,
+                            "in Json::Value::removeMember(): requires objectValue");
+        if (type() == nullValue)
+            return;
+
+        CZString actualKey(key, unsigned(strlen(key)), CZString::noDuplication);
+        value_.map_->erase(actualKey);
+    }
+
+    void Value::removeMember(const String &key) { removeMember(key.c_str()); }
+
+    bool Value::removeIndex(ArrayIndex index, Value *removed) {
+        if (type() != arrayValue) {
+            return false;
+        }
+        CZString key(index);
+        auto it = value_.map_->find(key);
+        if (it == value_.map_->end()) {
+            return false;
+        }
+        if (removed)
+            *removed = it->second;
+        ArrayIndex oldSize = size();
+        // shift left all items left, into the place of the "removed"
+        for (ArrayIndex i = index; i < (oldSize - 1); ++i) {
+            CZString keey(i);
+            (*value_.map_)[keey] = (*this)[i + 1];
+        }
+        // erase the last one ("leftover")
+        CZString keyLast(oldSize - 1);
+        auto itLast = value_.map_->find(keyLast);
+        value_.map_->erase(itLast);
+        return true;
+    }
+
+    bool Value::isMember(char const *begin, char const *end) const {
+        Value const *value = find(begin, end);
+        return nullptr != value;
+    }
+
+    bool Value::isMember(char const *key) const {
+        return isMember(key, key + strlen(key));
+    }
+
+    bool Value::isMember(String const &key) const {
+        return isMember(key.data(), key.data() + key.length());
+    }
+
+    Value::Members Value::getMemberNames() const {
+        JSON_ASSERT_MESSAGE(
+                type() == nullValue || type() == objectValue,
+                "in Json::Value::getMemberNames(), value must be objectValue");
+        if (type() == nullValue)
+            return Value::Members();
+        Members members;
+        members.reserve(value_.map_->size());
+        ObjectValues::const_iterator it = value_.map_->begin();
+        ObjectValues::const_iterator itEnd = value_.map_->end();
+        for (; it != itEnd; ++it) {
+            members.push_back(String((*it).first.data(), (*it).first.length()));
+        }
+        return members;
+    }
+
+    static bool IsIntegral(double d) {
+        double integral_part;
+        return modf(d, &integral_part) == 0.0;
+    }
+
+    bool Value::isNull() const { return type() == nullValue; }
+
+    bool Value::isBool() const { return type() == booleanValue; }
+
+    bool Value::isInt() const {
+        switch (type()) {
+            case intValue:
+#if defined(JSON_HAS_INT64)
+                return value_.int_ >= minInt && value_.int_ <= maxInt;
+#else
+                return true;
+#endif
+            case uintValue:
+                return value_.uint_ <= UInt(maxInt);
+            case realValue:
+                return value_.real_ >= minInt && value_.real_ <= maxInt &&
+                       IsIntegral(value_.real_);
+            default:
+                break;
+        }
+        return false;
+    }
+
+    bool Value::isUInt() const {
+        switch (type()) {
+            case intValue:
+#if defined(JSON_HAS_INT64)
+                return value_.int_ >= 0 && LargestUInt(value_.int_) <= LargestUInt(maxUInt);
+#else
+                return value_.int_ >= 0;
+#endif
+            case uintValue:
+#if defined(JSON_HAS_INT64)
+                return value_.uint_ <= maxUInt;
+#else
+                return true;
+#endif
+            case realValue:
+                return value_.real_ >= 0 && value_.real_ <= maxUInt &&
+                       IsIntegral(value_.real_);
+            default:
+                break;
+        }
+        return false;
+    }
+
+    bool Value::isInt64() const {
+#if defined(JSON_HAS_INT64)
+        switch (type()) {
+            case intValue:
+                return true;
+            case uintValue:
+                return value_.uint_ <= UInt64(maxInt64);
+            case realValue:
+                // Note that maxInt64 (= 2^63 - 1) is not exactly representable as a
+                // double, so double(maxInt64) will be rounded up to 2^63. Therefore we
+                // require the value to be strictly less than the limit.
+                return value_.real_ >= double(minInt64) &&
+                       value_.real_ < double(maxInt64) && IsIntegral(value_.real_);
+            default:
+                break;
+        }
+#endif // JSON_HAS_INT64
+        return false;
+    }
+
+    bool Value::isUInt64() const {
+#if defined(JSON_HAS_INT64)
+        switch (type()) {
+            case intValue:
+                return value_.int_ >= 0;
+            case uintValue:
+                return true;
+            case realValue:
+                // Note that maxUInt64 (= 2^64 - 1) is not exactly representable as a
+                // double, so double(maxUInt64) will be rounded up to 2^64. Therefore we
+                // require the value to be strictly less than the limit.
+                return value_.real_ >= 0 && value_.real_ < maxUInt64AsDouble &&
+                       IsIntegral(value_.real_);
+            default:
+                break;
+        }
+#endif // JSON_HAS_INT64
+        return false;
+    }
+
+    bool Value::isIntegral() const {
+        switch (type()) {
+            case intValue:
+            case uintValue:
+                return true;
+            case realValue:
+#if defined(JSON_HAS_INT64)
+                // Note that maxUInt64 (= 2^64 - 1) is not exactly representable as a
+                // double, so double(maxUInt64) will be rounded up to 2^64. Therefore we
+                // require the value to be strictly less than the limit.
+                return value_.real_ >= double(minInt64) &&
+                       value_.real_ < maxUInt64AsDouble && IsIntegral(value_.real_);
+#else
+                                                                                                                                        return value_.real_ >= minInt && value_.real_ <= maxUInt &&
+           IsIntegral(value_.real_);
+#endif // JSON_HAS_INT64
+            default:
+                break;
+        }
+        return false;
+    }
+
+    bool Value::isDouble() const {
+        return type() == intValue || type() == uintValue || type() == realValue;
+    }
+
+    bool Value::isNumeric() const { return isDouble(); }
+
+    bool Value::isString() const { return type() == stringValue; }
+
+    bool Value::isArray() const { return type() == arrayValue; }
+
+    bool Value::isObject() const { return type() == objectValue; }
+
+    Value::Comments::Comments(const Comments &that)
+            : ptr_{cloneUnique(that.ptr_)} {}
+
+    Value::Comments::Comments(Comments &&that) : ptr_{std::move(that.ptr_)} {}
+
+    Value::Comments &Value::Comments::operator=(const Comments &that) {
+        ptr_ = cloneUnique(that.ptr_);
+        return *this;
+    }
+
+    Value::Comments &Value::Comments::operator=(Comments &&that) {
+        ptr_ = std::move(that.ptr_);
+        return *this;
+    }
+
+    bool Value::Comments::has(CommentPlacement slot) const {
+        return ptr_ && !(*ptr_)[slot].empty();
+    }
+
+    String Value::Comments::get(CommentPlacement slot) const {
+        if (!ptr_)
+            return {};
+        return (*ptr_)[slot];
+    }
+
+    void Value::Comments::set(CommentPlacement slot, String comment) {
+        if (!ptr_) {
+            ptr_ = std::unique_ptr<Array>(new Array());
+        }
+        // check comments array boundry.
+        if (slot < CommentPlacement::numberOfCommentPlacement) {
+            (*ptr_)[slot] = std::move(comment);
+        }
+    }
+
+    void Value::setComment(String comment, CommentPlacement placement) {
+        if (!comment.empty() && (comment.back() == '\n')) {
+            // Always discard trailing newline, to aid indentation.
+            comment.pop_back();
+        }
+        JSON_ASSERT(!comment.empty());
+        JSON_ASSERT_MESSAGE(
+                comment[0] == '\0' || comment[0] == '/',
+                "in Json::Value::setComment(): Comments must start with /");
+        comments_.set(placement, std::move(comment));
+    }
+
+    bool Value::hasComment(CommentPlacement placement) const {
+        return comments_.has(placement);
+    }
+
+    String Value::getComment(CommentPlacement placement) const {
+        return comments_.get(placement);
+    }
+
+    void Value::setOffsetStart(ptrdiff_t start) { start_ = start; }
+
+    void Value::setOffsetLimit(ptrdiff_t limit) { limit_ = limit; }
+
+    ptrdiff_t Value::getOffsetStart() const { return start_; }
+
+    ptrdiff_t Value::getOffsetLimit() const { return limit_; }
+
+    String Value::toStyledString() const {
+        StreamWriterBuilder builder;
+
+        String out = this->hasComment(commentBefore) ? "\n" : "";
+        out += Json::writeString(builder, *this);
+        out += '\n';
+
+        return out;
+    }
+
+    Value::const_iterator Value::begin() const {
+        switch (type()) {
+            case arrayValue:
+            case objectValue:
+                if (value_.map_)
+                    return const_iterator(value_.map_->begin());
+                break;
+            default:
+                break;
+        }
+        return {};
+    }
+
+    Value::const_iterator Value::end() const {
+        switch (type()) {
+            case arrayValue:
+            case objectValue:
+                if (value_.map_)
+                    return const_iterator(value_.map_->end());
+                break;
+            default:
+                break;
+        }
+        return {};
+    }
+
+    Value::iterator Value::begin() {
+        switch (type()) {
+            case arrayValue:
+            case objectValue:
+                if (value_.map_)
+                    return iterator(value_.map_->begin());
+                break;
+            default:
+                break;
+        }
+        return iterator();
+    }
+
+    Value::iterator Value::end() {
+        switch (type()) {
+            case arrayValue:
+            case objectValue:
+                if (value_.map_)
+                    return iterator(value_.map_->end());
+                break;
+            default:
+                break;
+        }
+        return iterator();
+    }
+
+// class PathArgument
+// //////////////////////////////////////////////////////////////////
+
+    PathArgument::PathArgument() = default;
+
+    PathArgument::PathArgument(ArrayIndex index)
+            : index_(index), kind_(kindIndex) {}
+
+    PathArgument::PathArgument(const char *key) : key_(key), kind_(kindKey) {}
+
+    PathArgument::PathArgument(String key) : key_(std::move(key)), kind_(kindKey) {}
+
+// class Path
+// //////////////////////////////////////////////////////////////////
+
+    Path::Path(const String &path, const PathArgument &a1, const PathArgument &a2,
+               const PathArgument &a3, const PathArgument &a4,
+               const PathArgument &a5) {
+        InArgs in;
+        in.reserve(5);
+        in.push_back(&a1);
+        in.push_back(&a2);
+        in.push_back(&a3);
+        in.push_back(&a4);
+        in.push_back(&a5);
+        makePath(path, in);
+    }
+
+    void Path::makePath(const String &path, const InArgs &in) {
+        const char *current = path.c_str();
+        const char *end = current + path.length();
+        auto itInArg = in.begin();
+        while (current != end) {
+            if (*current == '[') {
+                ++current;
+                if (*current == '%')
+                    addPathInArg(path, in, itInArg, PathArgument::kindIndex);
+                else {
+                    ArrayIndex index = 0;
+                    for (; current != end && *current >= '0' && *current <= '9'; ++current)
+                        index = index * 10 + ArrayIndex(*current - '0');
+                    args_.push_back(index);
+                }
+                if (current == end || *++current != ']')
+                    invalidPath(path, int(current - path.c_str()));
+            } else if (*current == '%') {
+                addPathInArg(path, in, itInArg, PathArgument::kindKey);
+                ++current;
+            } else if (*current == '.' || *current == ']') {
+                ++current;
+            } else {
+                const char *beginName = current;
+                while (current != end && !strchr("[.", *current))
+                    ++current;
+                args_.push_back(String(beginName, current));
+            }
+        }
+    }
+
+    void Path::addPathInArg(const String & /*path*/, const InArgs &in,
+                            InArgs::const_iterator &itInArg,
+                            PathArgument::Kind kind) {
+        if (itInArg == in.end()) {
+            // Error: missing argument %d
+        } else if ((*itInArg)->kind_ != kind) {
+            // Error: bad argument type
+        } else {
+            args_.push_back(**itInArg++);
+        }
+    }
+
+    void Path::invalidPath(const String & /*path*/, int /*location*/) {
+        // Error: invalid path.
+    }
+
+    const Value &Path::resolve(const Value &root) const {
+        const Value *node = &root;
+        for (const auto &arg : args_) {
+            if (arg.kind_ == PathArgument::kindIndex) {
+                if (!node->isArray() || !node->isValidIndex(arg.index_)) {
+                    // Error: unable to resolve path (array value expected at position... )
+                    return Value::nullSingleton();
+                }
+                node = &((*node)[arg.index_]);
+            } else if (arg.kind_ == PathArgument::kindKey) {
+                if (!node->isObject()) {
+                    // Error: unable to resolve path (object value expected at position...)
+                    return Value::nullSingleton();
+                }
+                node = &((*node)[arg.key_]);
+                if (node == &Value::nullSingleton()) {
+                    // Error: unable to resolve path (object has no member named '' at
+                    // position...)
+                    return Value::nullSingleton();
+                }
+            }
+        }
+        return *node;
+    }
+
+    Value Path::resolve(const Value &root, const Value &defaultValue) const {
+        const Value *node = &root;
+        for (const auto &arg : args_) {
+            if (arg.kind_ == PathArgument::kindIndex) {
+                if (!node->isArray() || !node->isValidIndex(arg.index_))
+                    return defaultValue;
+                node = &((*node)[arg.index_]);
+            } else if (arg.kind_ == PathArgument::kindKey) {
+                if (!node->isObject())
+                    return defaultValue;
+                node = &((*node)[arg.key_]);
+                if (node == &Value::nullSingleton())
+                    return defaultValue;
+            }
+        }
+        return *node;
+    }
+
+    Value &Path::make(Value &root) const {
+        Value *node = &root;
+        for (const auto &arg : args_) {
+            if (arg.kind_ == PathArgument::kindIndex) {
+                if (!node->isArray()) {
+                    // Error: node is not an array at position ...
+                }
+                node = &((*node)[arg.index_]);
+            } else if (arg.kind_ == PathArgument::kindKey) {
+                if (!node->isObject()) {
+                    // Error: node is not an object at position...
+                }
+                node = &((*node)[arg.key_]);
+            }
+        }
+        return *node;
+    }
+
+} // namespace Json
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: src/lib_json/json_value.cpp
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+
+
+// //////////////////////////////////////////////////////////////////////
+// Beginning of content of file: src/lib_json/json_writer.cpp
+// //////////////////////////////////////////////////////////////////////
+
+// Copyright 2011 Baptiste Lepilleur and The JsonCpp Authors
+// Distributed under MIT license, or public domain if desired and
+// recognized in your jurisdiction.
+// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
+
+#if !defined(JSON_IS_AMALGAMATION)
+                                                                                                                        #include "json_tool.h"
+#include <json/writer.h>
+#endif // if !defined(JSON_IS_AMALGAMATION)
+
+#include <algorithm>
+#include <cassert>
+#include <cctype>
+#include <cstring>
+#include <iomanip>
+#include <memory>
+#include <set>
+#include <sstream>
+#include <utility>
+
+#if __cplusplus >= 201103L
+
+#include <cmath>
+#include <cstdio>
+
+#if !defined(isnan)
+#define isnan std::isnan
+#endif
+
+#if !defined(isfinite)
+#define isfinite std::isfinite
+#endif
+
+#else
+                                                                                                                        #include <cmath>
+#include <cstdio>
+
+#if defined(_MSC_VER)
+#if !defined(isnan)
+#include <float.h>
+#define isnan _isnan
+#endif
+
+#if !defined(isfinite)
+#include <float.h>
+#define isfinite _finite
+#endif
+
+#if !defined(_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES)
+#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1
+#endif //_CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES
+
+#endif //_MSC_VER
+
+#if defined(__sun) && defined(__SVR4) // Solaris
+#if !defined(isfinite)
+#include <ieeefp.h>
+#define isfinite finite
+#endif
+#endif
+
+#if defined(__hpux)
+#if !defined(isfinite)
+#if defined(__ia64) && !defined(finite)
+#define isfinite(x)                                                            \
+  ((sizeof(x) == sizeof(float) ? _Isfinitef(x) : _IsFinite(x)))
+#endif
+#endif
+#endif
+
+#if !defined(isnan)
+// IEEE standard states that NaN values will not compare to themselves
+#define isnan(x) (x != x)
+#endif
+
+#if !defined(__APPLE__)
+#if !defined(isfinite)
+#define isfinite finite
+#endif
+#endif
+#endif
+
+#if defined(_MSC_VER)
+                                                                                                                        // Disable warning about strdup being deprecated.
+#pragma warning(disable : 4996)
+#endif
+
+namespace Json {
+
+#if __cplusplus >= 201103L || (defined(_CPPLIB_VER) && _CPPLIB_VER >= 520)
+    using StreamWriterPtr = std::unique_ptr<StreamWriter>;
+#else
+    using StreamWriterPtr = std::auto_ptr<StreamWriter>;
+#endif
+
+    String valueToString(LargestInt value) {
+        UIntToStringBuffer buffer;
+        char *current = buffer + sizeof(buffer);
+        if (value == Value::minLargestInt) {
+            uintToString(LargestUInt(Value::maxLargestInt) + 1, current);
+            *--current = '-';
+        } else if (value < 0) {
+            uintToString(LargestUInt(-value), current);
+            *--current = '-';
+        } else {
+            uintToString(LargestUInt(value), current);
+        }
+        assert(current >= buffer);
+        return current;
+    }
+
+    String valueToString(LargestUInt value) {
+        UIntToStringBuffer buffer;
+        char *current = buffer + sizeof(buffer);
+        uintToString(value, current);
+        assert(current >= buffer);
+        return current;
+    }
+
+#if defined(JSON_HAS_INT64)
+
+    String valueToString(Int value) { return valueToString(LargestInt(value)); }
+
+    String valueToString(UInt value) { return valueToString(LargestUInt(value)); }
+
+#endif // # if defined(JSON_HAS_INT64)
+
+    namespace {
+        String valueToString(double value, bool useSpecialFloats,
+                             unsigned int precision, PrecisionType precisionType) {
+            // Print into the buffer. We need not request the alternative representation
+            // that always has a decimal point because JSON doesn't distinguish the
+            // concepts of reals and integers.
+            if (!isfinite(value)) {
+                static const char *const reps[2][3] = {{"NaN",  "-Infinity", "Infinity"},
+                                                       {"null", "-1e+9999",  "1e+9999"}};
+                return reps[useSpecialFloats ? 0 : 1]
+                [isnan(value) ? 0 : (value < 0) ? 1 : 2];
+            }
+
+            String buffer(size_t(36), '\0');
+            while (true) {
+                int len = jsoncpp_snprintf(
+                        &*buffer.begin(), buffer.size(),
+                        (precisionType == PrecisionType::significantDigits) ? "%.*g" : "%.*f",
+                        precision, value);
+                assert(len >= 0);
+                auto wouldPrint = static_cast<size_t>(len);
+                if (wouldPrint >= buffer.size()) {
+                    buffer.resize(wouldPrint + 1);
+                    continue;
+                }
+                buffer.resize(wouldPrint);
+                break;
+            }
+
+            buffer.erase(fixNumericLocale(buffer.begin(), buffer.end()), buffer.end());
+
+            // strip the zero padding from the right
+            if (precisionType == PrecisionType::decimalPlaces) {
+                buffer.erase(fixZerosInTheEnd(buffer.begin(), buffer.end()), buffer.end());
+            }
+
+            // try to ensure we preserve the fact that this was given to us as a double on
+            // input
+            if (buffer.find('.') == buffer.npos && buffer.find('e') == buffer.npos) {
+                buffer += ".0";
+            }
+            return buffer;
+        }
+    } // namespace
+
+    String valueToString(double value, unsigned int precision,
+                         PrecisionType precisionType) {
+        return valueToString(value, false, precision, precisionType);
+    }
+
+    String valueToString(bool value) { return value ? "true" : "false"; }
+
+    static bool doesAnyCharRequireEscaping(char const *s, size_t n) {
+        assert(s || !n);
+
+        return std::any_of(s, s + n, [](unsigned char c) {
+            return c == '\\' || c == '"' || c < 0x20 || c > 0x7F;
+        });
+    }
+
+    static unsigned int utf8ToCodepoint(const char *&s, const char *e) {
+        const unsigned int REPLACEMENT_CHARACTER = 0xFFFD;
+
+        unsigned int firstByte = static_cast<unsigned char>(*s);
+
+        if (firstByte < 0x80)
+            return firstByte;
+
+        if (firstByte < 0xE0) {
+            if (e - s < 2)
+                return REPLACEMENT_CHARACTER;
+
+            unsigned int calculated =
+                    ((firstByte & 0x1F) << 6) | (static_cast<unsigned int>(s[1]) & 0x3F);
+            s += 1;
+            // oversized encoded characters are invalid
+            return calculated < 0x80 ? REPLACEMENT_CHARACTER : calculated;
+        }
+
+        if (firstByte < 0xF0) {
+            if (e - s < 3)
+                return REPLACEMENT_CHARACTER;
+
+            unsigned int calculated = ((firstByte & 0x0F) << 12) |
+                                      ((static_cast<unsigned int>(s[1]) & 0x3F) << 6) |
+                                      (static_cast<unsigned int>(s[2]) & 0x3F);
+            s += 2;
+            // surrogates aren't valid codepoints itself
+            // shouldn't be UTF-8 encoded
+            if (calculated >= 0xD800 && calculated <= 0xDFFF)
+                return REPLACEMENT_CHARACTER;
+            // oversized encoded characters are invalid
+            return calculated < 0x800 ? REPLACEMENT_CHARACTER : calculated;
+        }
+
+        if (firstByte < 0xF8) {
+            if (e - s < 4)
+                return REPLACEMENT_CHARACTER;
+
+            unsigned int calculated = ((firstByte & 0x07) << 18) |
+                                      ((static_cast<unsigned int>(s[1]) & 0x3F) << 12) |
+                                      ((static_cast<unsigned int>(s[2]) & 0x3F) << 6) |
+                                      (static_cast<unsigned int>(s[3]) & 0x3F);
+            s += 3;
+            // oversized encoded characters are invalid
+            return calculated < 0x10000 ? REPLACEMENT_CHARACTER : calculated;
+        }
+
+        return REPLACEMENT_CHARACTER;
+    }
+
+    static const char hex2[] = "000102030405060708090a0b0c0d0e0f"
+                               "101112131415161718191a1b1c1d1e1f"
+                               "202122232425262728292a2b2c2d2e2f"
+                               "303132333435363738393a3b3c3d3e3f"
+                               "404142434445464748494a4b4c4d4e4f"
+                               "505152535455565758595a5b5c5d5e5f"
+                               "606162636465666768696a6b6c6d6e6f"
+                               "707172737475767778797a7b7c7d7e7f"
+                               "808182838485868788898a8b8c8d8e8f"
+                               "909192939495969798999a9b9c9d9e9f"
+                               "a0a1a2a3a4a5a6a7a8a9aaabacadaeaf"
+                               "b0b1b2b3b4b5b6b7b8b9babbbcbdbebf"
+                               "c0c1c2c3c4c5c6c7c8c9cacbcccdcecf"
+                               "d0d1d2d3d4d5d6d7d8d9dadbdcdddedf"
+                               "e0e1e2e3e4e5e6e7e8e9eaebecedeeef"
+                               "f0f1f2f3f4f5f6f7f8f9fafbfcfdfeff";
+
+    static String toHex16Bit(unsigned int x) {
+        const unsigned int hi = (x >> 8) & 0xff;
+        const unsigned int lo = x & 0xff;
+        String result(4, ' ');
+        result[0] = hex2[2 * hi];
+        result[1] = hex2[2 * hi + 1];
+        result[2] = hex2[2 * lo];
+        result[3] = hex2[2 * lo + 1];
+        return result;
+    }
+
+    static void appendRaw(String &result, unsigned ch) {
+        result += static_cast<char>(ch);
+    }
+
+    static void appendHex(String &result, unsigned ch) {
+        result.append("\\u").append(toHex16Bit(ch));
+    }
+
+    static String valueToQuotedStringN(const char *value, unsigned length,
+                                       bool emitUTF8 = false) {
+        if (value == nullptr)
+            return "";
+
+        if (!doesAnyCharRequireEscaping(value, length))
+            return String("\"") + value + "\"";
+        // We have to walk value and escape any special characters.
+        // Appending to String is not efficient, but this should be rare.
+        // (Note: forward slashes are *not* rare, but I am not escaping them.)
+        String::size_type maxsize = length * 2 + 3; // allescaped+quotes+NULL
+        String result;
+        result.reserve(maxsize); // to avoid lots of mallocs
+        result += "\"";
+        char const *end = value + length;
+        for (const char *c = value; c != end; ++c) {
+            switch (*c) {
+                case '\"':
+                    result += "\\\"";
+                    break;
+                case '\\':
+                    result += "\\\\";
+                    break;
+                case '\b':
+                    result += "\\b";
+                    break;
+                case '\f':
+                    result += "\\f";
+                    break;
+                case '\n':
+                    result += "\\n";
+                    break;
+                case '\r':
+                    result += "\\r";
+                    break;
+                case '\t':
+                    result += "\\t";
+                    break;
+                    // case '/':
+                    // Even though \/ is considered a legal escape in JSON, a bare
+                    // slash is also legal, so I see no reason to escape it.
+                    // (I hope I am not misunderstanding something.)
+                    // blep notes: actually escaping \/ may be useful in javascript to avoid </
+                    // sequence.
+                    // Should add a flag to allow this compatibility mode and prevent this
+                    // sequence from occurring.
+                default: {
+                    if (emitUTF8) {
+                        unsigned codepoint = static_cast<unsigned char>(*c);
+                        if (codepoint < 0x20) {
+                            appendHex(result, codepoint);
+                        } else {
+                            appendRaw(result, codepoint);
+                        }
+                    } else {
+                        unsigned codepoint = utf8ToCodepoint(c, end); // modifies `c`
+                        if (codepoint < 0x20) {
+                            appendHex(result, codepoint);
+                        } else if (codepoint < 0x80) {
+                            appendRaw(result, codepoint);
+                        } else if (codepoint < 0x10000) {
+                            // Basic Multilingual Plane
+                            appendHex(result, codepoint);
+                        } else {
+                            // Extended Unicode. Encode 20 bits as a surrogate pair.
+                            codepoint -= 0x10000;
+                            appendHex(result, 0xd800 + ((codepoint >> 10) & 0x3ff));
+                            appendHex(result, 0xdc00 + (codepoint & 0x3ff));
+                        }
+                    }
+                }
+                    break;
+            }
+        }
+        result += "\"";
+        return result;
+    }
+
+    String valueToQuotedString(const char *value) {
+        return valueToQuotedStringN(value, static_cast<unsigned int>(strlen(value)));
+    }
+
+// Class Writer
+// //////////////////////////////////////////////////////////////////
+    Writer::~Writer() = default;
+
+// Class FastWriter
+// //////////////////////////////////////////////////////////////////
+
+    FastWriter::FastWriter()
+
+    = default;
+
+    void FastWriter::enableYAMLCompatibility() { yamlCompatibilityEnabled_ = true; }
+
+    void FastWriter::dropNullPlaceholders() { dropNullPlaceholders_ = true; }
+
+    void FastWriter::omitEndingLineFeed() { omitEndingLineFeed_ = true; }
+
+    String FastWriter::write(const Value &root) {
+        document_.clear();
+        writeValue(root);
+        if (!omitEndingLineFeed_)
+            document_ += '\n';
+        return document_;
+    }
+
+    void FastWriter::writeValue(const Value &value) {
+        switch (value.type()) {
+            case nullValue:
+                if (!dropNullPlaceholders_)
+                    document_ += "null";
+                break;
+            case intValue:
+                document_ += valueToString(value.asLargestInt());
+                break;
+            case uintValue:
+                document_ += valueToString(value.asLargestUInt());
+                break;
+            case realValue:
+                document_ += valueToString(value.asDouble());
+                break;
+            case stringValue: {
+                // Is NULL possible for value.string_? No.
+                char const *str;
+                char const *end;
+                bool ok = value.getString(&str, &end);
+                if (ok)
+                    document_ += valueToQuotedStringN(str, static_cast<unsigned>(end - str));
+                break;
+            }
+            case booleanValue:
+                document_ += valueToString(value.asBool());
+                break;
+            case arrayValue: {
+                document_ += '[';
+                ArrayIndex size = value.size();
+                for (ArrayIndex index = 0; index < size; ++index) {
+                    if (index > 0)
+                        document_ += ',';
+                    writeValue(value[index]);
+                }
+                document_ += ']';
+            }
+                break;
+            case objectValue: {
+                Value::Members members(value.getMemberNames());
+                document_ += '{';
+                for (auto it = members.begin(); it != members.end(); ++it) {
+                    const String &name = *it;
+                    if (it != members.begin())
+                        document_ += ',';
+                    document_ += valueToQuotedStringN(name.data(),
+                                                      static_cast<unsigned>(name.length()));
+                    document_ += yamlCompatibilityEnabled_ ? ": " : ":";
+                    writeValue(value[name]);
+                }
+                document_ += '}';
+            }
+                break;
+        }
+    }
+
+// Class StyledWriter
+// //////////////////////////////////////////////////////////////////
+
+    StyledWriter::StyledWriter() = default;
+
+    String StyledWriter::write(const Value &root) {
+        document_.clear();
+        addChildValues_ = false;
+        indentString_.clear();
+        writeCommentBeforeValue(root);
+        writeValue(root);
+        writeCommentAfterValueOnSameLine(root);
+        document_ += '\n';
+        return document_;
+    }
+
+    void StyledWriter::writeValue(const Value &value) {
+        switch (value.type()) {
+            case nullValue:
+                pushValue("null");
+                break;
+            case intValue:
+                pushValue(valueToString(value.asLargestInt()));
+                break;
+            case uintValue:
+                pushValue(valueToString(value.asLargestUInt()));
+                break;
+            case realValue:
+                pushValue(valueToString(value.asDouble()));
+                break;
+            case stringValue: {
+                // Is NULL possible for value.string_? No.
+                char const *str;
+                char const *end;
+                bool ok = value.getString(&str, &end);
+                if (ok)
+                    pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end - str)));
+                else
+                    pushValue("");
+                break;
+            }
+            case booleanValue:
+                pushValue(valueToString(value.asBool()));
+                break;
+            case arrayValue:
+                writeArrayValue(value);
+                break;
+            case objectValue: {
+                Value::Members members(value.getMemberNames());
+                if (members.empty())
+                    pushValue("{}");
+                else {
+                    writeWithIndent("{");
+                    indent();
+                    auto it = members.begin();
+                    for (;;) {
+                        const String &name = *it;
+                        const Value &childValue = value[name];
+                        writeCommentBeforeValue(childValue);
+                        writeWithIndent(valueToQuotedString(name.c_str()));
+                        document_ += " : ";
+                        writeValue(childValue);
+                        if (++it == members.end()) {
+                            writeCommentAfterValueOnSameLine(childValue);
+                            break;
+                        }
+                        document_ += ',';
+                        writeCommentAfterValueOnSameLine(childValue);
+                    }
+                    unindent();
+                    writeWithIndent("}");
+                }
+            }
+                break;
+        }
+    }
+
+    void StyledWriter::writeArrayValue(const Value &value) {
+        unsigned size = value.size();
+        if (size == 0)
+            pushValue("[]");
+        else {
+            bool isArrayMultiLine = isMultilineArray(value);
+            if (isArrayMultiLine) {
+                writeWithIndent("[");
+                indent();
+                bool hasChildValue = !childValues_.empty();
+                unsigned index = 0;
+                for (;;) {
+                    const Value &childValue = value[index];
+                    writeCommentBeforeValue(childValue);
+                    if (hasChildValue)
+                        writeWithIndent(childValues_[index]);
+                    else {
+                        writeIndent();
+                        writeValue(childValue);
+                    }
+                    if (++index == size) {
+                        writeCommentAfterValueOnSameLine(childValue);
+                        break;
+                    }
+                    document_ += ',';
+                    writeCommentAfterValueOnSameLine(childValue);
+                }
+                unindent();
+                writeWithIndent("]");
+            } else // output on a single line
+            {
+                assert(childValues_.size() == size);
+                document_ += "[ ";
+                for (unsigned index = 0; index < size; ++index) {
+                    if (index > 0)
+                        document_ += ", ";
+                    document_ += childValues_[index];
+                }
+                document_ += " ]";
+            }
+        }
+    }
+
+    bool StyledWriter::isMultilineArray(const Value &value) {
+        ArrayIndex const size = value.size();
+        bool isMultiLine = size * 3 >= rightMargin_;
+        childValues_.clear();
+        for (ArrayIndex index = 0; index < size && !isMultiLine; ++index) {
+            const Value &childValue = value[index];
+            isMultiLine = ((childValue.isArray() || childValue.isObject()) &&
+                           !childValue.empty());
+        }
+        if (!isMultiLine) // check if line length > max line length
+        {
+            childValues_.reserve(size);
+            addChildValues_ = true;
+            ArrayIndex lineLength = 4 + (size - 1) * 2; // '[ ' + ', '*n + ' ]'
+            for (ArrayIndex index = 0; index < size; ++index) {
+                if (hasCommentForValue(value[index])) {
+                    isMultiLine = true;
+                }
+                writeValue(value[index]);
+                lineLength += static_cast<ArrayIndex>(childValues_[index].length());
+            }
+            addChildValues_ = false;
+            isMultiLine = isMultiLine || lineLength >= rightMargin_;
+        }
+        return isMultiLine;
+    }
+
+    void StyledWriter::pushValue(const String &value) {
+        if (addChildValues_)
+            childValues_.push_back(value);
+        else
+            document_ += value;
+    }
+
+    void StyledWriter::writeIndent() {
+        if (!document_.empty()) {
+            char last = document_[document_.length() - 1];
+            if (last == ' ') // already indented
+                return;
+            if (last != '\n') // Comments may add new-line
+                document_ += '\n';
+        }
+        document_ += indentString_;
+    }
+
+    void StyledWriter::writeWithIndent(const String &value) {
+        writeIndent();
+        document_ += value;
+    }
+
+    void StyledWriter::indent() { indentString_ += String(indentSize_, ' '); }
+
+    void StyledWriter::unindent() {
+        assert(indentString_.size() >= indentSize_);
+        indentString_.resize(indentString_.size() - indentSize_);
+    }
+
+    void StyledWriter::writeCommentBeforeValue(const Value &root) {
+        if (!root.hasComment(commentBefore))
+            return;
+
+        document_ += '\n';
+        writeIndent();
+        const String &comment = root.getComment(commentBefore);
+        String::const_iterator iter = comment.begin();
+        while (iter != comment.end()) {
+            document_ += *iter;
+            if (*iter == '\n' && ((iter + 1) != comment.end() && *(iter + 1) == '/'))
+                writeIndent();
+            ++iter;
+        }
+
+        // Comments are stripped of trailing newlines, so add one here
+        document_ += '\n';
+    }
+
+    void StyledWriter::writeCommentAfterValueOnSameLine(const Value &root) {
+        if (root.hasComment(commentAfterOnSameLine))
+            document_ += " " + root.getComment(commentAfterOnSameLine);
+
+        if (root.hasComment(commentAfter)) {
+            document_ += '\n';
+            document_ += root.getComment(commentAfter);
+            document_ += '\n';
+        }
+    }
+
+    bool StyledWriter::hasCommentForValue(const Value &value) {
+        return value.hasComment(commentBefore) ||
+               value.hasComment(commentAfterOnSameLine) ||
+               value.hasComment(commentAfter);
+    }
+
+// Class StyledStreamWriter
+// //////////////////////////////////////////////////////////////////
+
+    StyledStreamWriter::StyledStreamWriter(String indentation)
+            : document_(nullptr), indentation_(std::move(indentation)),
+              addChildValues_(), indented_(false) {}
+
+    void StyledStreamWriter::write(OStream &out, const Value &root) {
+        document_ = &out;
+        addChildValues_ = false;
+        indentString_.clear();
+        indented_ = true;
+        writeCommentBeforeValue(root);
+        if (!indented_)
+            writeIndent();
+        indented_ = true;
+        writeValue(root);
+        writeCommentAfterValueOnSameLine(root);
+        *document_ << "\n";
+        document_ = nullptr; // Forget the stream, for safety.
+    }
+
+    void StyledStreamWriter::writeValue(const Value &value) {
+        switch (value.type()) {
+            case nullValue:
+                pushValue("null");
+                break;
+            case intValue:
+                pushValue(valueToString(value.asLargestInt()));
+                break;
+            case uintValue:
+                pushValue(valueToString(value.asLargestUInt()));
+                break;
+            case realValue:
+                pushValue(valueToString(value.asDouble()));
+                break;
+            case stringValue: {
+                // Is NULL possible for value.string_? No.
+                char const *str;
+                char const *end;
+                bool ok = value.getString(&str, &end);
+                if (ok)
+                    pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end - str)));
+                else
+                    pushValue("");
+                break;
+            }
+            case booleanValue:
+                pushValue(valueToString(value.asBool()));
+                break;
+            case arrayValue:
+                writeArrayValue(value);
+                break;
+            case objectValue: {
+                Value::Members members(value.getMemberNames());
+                if (members.empty())
+                    pushValue("{}");
+                else {
+                    writeWithIndent("{");
+                    indent();
+                    auto it = members.begin();
+                    for (;;) {
+                        const String &name = *it;
+                        const Value &childValue = value[name];
+                        writeCommentBeforeValue(childValue);
+                        writeWithIndent(valueToQuotedString(name.c_str()));
+                        *document_ << " : ";
+                        writeValue(childValue);
+                        if (++it == members.end()) {
+                            writeCommentAfterValueOnSameLine(childValue);
+                            break;
+                        }
+                        *document_ << ",";
+                        writeCommentAfterValueOnSameLine(childValue);
+                    }
+                    unindent();
+                    writeWithIndent("}");
+                }
+            }
+                break;
+        }
+    }
+
+    void StyledStreamWriter::writeArrayValue(const Value &value) {
+        unsigned size = value.size();
+        if (size == 0)
+            pushValue("[]");
+        else {
+            bool isArrayMultiLine = isMultilineArray(value);
+            if (isArrayMultiLine) {
+                writeWithIndent("[");
+                indent();
+                bool hasChildValue = !childValues_.empty();
+                unsigned index = 0;
+                for (;;) {
+                    const Value &childValue = value[index];
+                    writeCommentBeforeValue(childValue);
+                    if (hasChildValue)
+                        writeWithIndent(childValues_[index]);
+                    else {
+                        if (!indented_)
+                            writeIndent();
+                        indented_ = true;
+                        writeValue(childValue);
+                        indented_ = false;
+                    }
+                    if (++index == size) {
+                        writeCommentAfterValueOnSameLine(childValue);
+                        break;
+                    }
+                    *document_ << ",";
+                    writeCommentAfterValueOnSameLine(childValue);
+                }
+                unindent();
+                writeWithIndent("]");
+            } else // output on a single line
+            {
+                assert(childValues_.size() == size);
+                *document_ << "[ ";
+                for (unsigned index = 0; index < size; ++index) {
+                    if (index > 0)
+                        *document_ << ", ";
+                    *document_ << childValues_[index];
+                }
+                *document_ << " ]";
+            }
+        }
+    }
+
+    bool StyledStreamWriter::isMultilineArray(const Value &value) {
+        ArrayIndex const size = value.size();
+        bool isMultiLine = size * 3 >= rightMargin_;
+        childValues_.clear();
+        for (ArrayIndex index = 0; index < size && !isMultiLine; ++index) {
+            const Value &childValue = value[index];
+            isMultiLine = ((childValue.isArray() || childValue.isObject()) &&
+                           !childValue.empty());
+        }
+        if (!isMultiLine) // check if line length > max line length
+        {
+            childValues_.reserve(size);
+            addChildValues_ = true;
+            ArrayIndex lineLength = 4 + (size - 1) * 2; // '[ ' + ', '*n + ' ]'
+            for (ArrayIndex index = 0; index < size; ++index) {
+                if (hasCommentForValue(value[index])) {
+                    isMultiLine = true;
+                }
+                writeValue(value[index]);
+                lineLength += static_cast<ArrayIndex>(childValues_[index].length());
+            }
+            addChildValues_ = false;
+            isMultiLine = isMultiLine || lineLength >= rightMargin_;
+        }
+        return isMultiLine;
+    }
+
+    void StyledStreamWriter::pushValue(const String &value) {
+        if (addChildValues_)
+            childValues_.push_back(value);
+        else
+            *document_ << value;
+    }
+
+    void StyledStreamWriter::writeIndent() {
+        // blep intended this to look at the so-far-written string
+        // to determine whether we are already indented, but
+        // with a stream we cannot do that. So we rely on some saved state.
+        // The caller checks indented_.
+        *document_ << '\n' << indentString_;
+    }
+
+    void StyledStreamWriter::writeWithIndent(const String &value) {
+        if (!indented_)
+            writeIndent();
+        *document_ << value;
+        indented_ = false;
+    }
+
+    void StyledStreamWriter::indent() { indentString_ += indentation_; }
+
+    void StyledStreamWriter::unindent() {
+        assert(indentString_.size() >= indentation_.size());
+        indentString_.resize(indentString_.size() - indentation_.size());
+    }
+
+    void StyledStreamWriter::writeCommentBeforeValue(const Value &root) {
+        if (!root.hasComment(commentBefore))
+            return;
+
+        if (!indented_)
+            writeIndent();
+        const String &comment = root.getComment(commentBefore);
+        String::const_iterator iter = comment.begin();
+        while (iter != comment.end()) {
+            *document_ << *iter;
+            if (*iter == '\n' && ((iter + 1) != comment.end() && *(iter + 1) == '/'))
+                // writeIndent();  // would include newline
+                *document_ << indentString_;
+            ++iter;
+        }
+        indented_ = false;
+    }
+
+    void StyledStreamWriter::writeCommentAfterValueOnSameLine(const Value &root) {
+        if (root.hasComment(commentAfterOnSameLine))
+            *document_ << ' ' << root.getComment(commentAfterOnSameLine);
+
+        if (root.hasComment(commentAfter)) {
+            writeIndent();
+            *document_ << root.getComment(commentAfter);
+        }
+        indented_ = false;
+    }
+
+    bool StyledStreamWriter::hasCommentForValue(const Value &value) {
+        return value.hasComment(commentBefore) ||
+               value.hasComment(commentAfterOnSameLine) ||
+               value.hasComment(commentAfter);
+    }
+
+//////////////////////////
+// BuiltStyledStreamWriter
+
+/// Scoped enums are not available until C++11.
+    struct CommentStyle {
+        /// Decide whether to write comments.
+        enum Enum {
+            None, ///< Drop all comments.
+            Most, ///< Recover odd behavior of previous versions (not implemented yet).
+            All   ///< Keep all comments.
+        };
+    };
+
+    struct BuiltStyledStreamWriter : public StreamWriter {
+        BuiltStyledStreamWriter(String indentation, CommentStyle::Enum cs,
+                                String colonSymbol, String nullSymbol,
+                                String endingLineFeedSymbol, bool useSpecialFloats,
+                                bool emitUTF8, unsigned int precision,
+                                PrecisionType precisionType);
+
+        int write(Value const &root, OStream *sout) override;
+
+    private:
+        void writeValue(Value const &value);
+
+        void writeArrayValue(Value const &value);
+
+        bool isMultilineArray(Value const &value);
+
+        void pushValue(String const &value);
+
+        void writeIndent();
+
+        void writeWithIndent(String const &value);
+
+        void indent();
+
+        void unindent();
+
+        void writeCommentBeforeValue(Value const &root);
+
+        void writeCommentAfterValueOnSameLine(Value const &root);
+
+        static bool hasCommentForValue(const Value &value);
+
+        using ChildValues = std::vector<String>;
+
+        ChildValues childValues_;
+        String indentString_;
+        unsigned int rightMargin_;
+        String indentation_;
+        CommentStyle::Enum cs_;
+        String colonSymbol_;
+        String nullSymbol_;
+        String endingLineFeedSymbol_;
+        bool addChildValues_: 1;
+        bool indented_: 1;
+        bool useSpecialFloats_: 1;
+        bool emitUTF8_: 1;
+        unsigned int precision_;
+        PrecisionType precisionType_;
+    };
+
+    BuiltStyledStreamWriter::BuiltStyledStreamWriter(
+            String indentation, CommentStyle::Enum cs, String colonSymbol,
+            String nullSymbol, String endingLineFeedSymbol, bool useSpecialFloats,
+            bool emitUTF8, unsigned int precision, PrecisionType precisionType)
+            : rightMargin_(74), indentation_(std::move(indentation)), cs_(cs),
+              colonSymbol_(std::move(colonSymbol)), nullSymbol_(std::move(nullSymbol)),
+              endingLineFeedSymbol_(std::move(endingLineFeedSymbol)),
+              addChildValues_(false), indented_(false),
+              useSpecialFloats_(useSpecialFloats), emitUTF8_(emitUTF8),
+              precision_(precision), precisionType_(precisionType) {}
+
+    int BuiltStyledStreamWriter::write(Value const &root, OStream *sout) {
+        sout_ = sout;
+        addChildValues_ = false;
+        indented_ = true;
+        indentString_.clear();
+        writeCommentBeforeValue(root);
+        if (!indented_)
+            writeIndent();
+        indented_ = true;
+        writeValue(root);
+        writeCommentAfterValueOnSameLine(root);
+        *sout_ << endingLineFeedSymbol_;
+        sout_ = nullptr;
+        return 0;
+    }
+
+    void BuiltStyledStreamWriter::writeValue(Value const &value) {
+        switch (value.type()) {
+            case nullValue:
+                pushValue(nullSymbol_);
+                break;
+            case intValue:
+                pushValue(valueToString(value.asLargestInt()));
+                break;
+            case uintValue:
+                pushValue(valueToString(value.asLargestUInt()));
+                break;
+            case realValue:
+                pushValue(valueToString(value.asDouble(), useSpecialFloats_, precision_,
+                                        precisionType_));
+                break;
+            case stringValue: {
+                // Is NULL is possible for value.string_? No.
+                char const *str;
+                char const *end;
+                bool ok = value.getString(&str, &end);
+                if (ok)
+                    pushValue(valueToQuotedStringN(str, static_cast<unsigned>(end - str),
+                                                   emitUTF8_));
+                else
+                    pushValue("");
+                break;
+            }
+            case booleanValue:
+                pushValue(valueToString(value.asBool()));
+                break;
+            case arrayValue:
+                writeArrayValue(value);
+                break;
+            case objectValue: {
+                Value::Members members(value.getMemberNames());
+                if (members.empty())
+                    pushValue("{}");
+                else {
+                    writeWithIndent("{");
+                    indent();
+                    auto it = members.begin();
+                    for (;;) {
+                        String const &name = *it;
+                        Value const &childValue = value[name];
+                        writeCommentBeforeValue(childValue);
+                        writeWithIndent(valueToQuotedStringN(
+                                name.data(), static_cast<unsigned>(name.length()), emitUTF8_));
+                        *sout_ << colonSymbol_;
+                        writeValue(childValue);
+                        if (++it == members.end()) {
+                            writeCommentAfterValueOnSameLine(childValue);
+                            break;
+                        }
+                        *sout_ << ",";
+                        writeCommentAfterValueOnSameLine(childValue);
+                    }
+                    unindent();
+                    writeWithIndent("}");
+                }
+            }
+                break;
+        }
+    }
+
+    void BuiltStyledStreamWriter::writeArrayValue(Value const &value) {
+        unsigned size = value.size();
+        if (size == 0)
+            pushValue("[]");
+        else {
+            bool isMultiLine = (cs_ == CommentStyle::All) || isMultilineArray(value);
+            if (isMultiLine) {
+                writeWithIndent("[");
+                indent();
+                bool hasChildValue = !childValues_.empty();
+                unsigned index = 0;
+                for (;;) {
+                    Value const &childValue = value[index];
+                    writeCommentBeforeValue(childValue);
+                    if (hasChildValue)
+                        writeWithIndent(childValues_[index]);
+                    else {
+                        if (!indented_)
+                            writeIndent();
+                        indented_ = true;
+                        writeValue(childValue);
+                        indented_ = false;
+                    }
+                    if (++index == size) {
+                        writeCommentAfterValueOnSameLine(childValue);
+                        break;
+                    }
+                    *sout_ << ",";
+                    writeCommentAfterValueOnSameLine(childValue);
+                }
+                unindent();
+                writeWithIndent("]");
+            } else // output on a single line
+            {
+                assert(childValues_.size() == size);
+                *sout_ << "[";
+                if (!indentation_.empty())
+                    *sout_ << " ";
+                for (unsigned index = 0; index < size; ++index) {
+                    if (index > 0)
+                        *sout_ << ((!indentation_.empty()) ? ", " : ",");
+                    *sout_ << childValues_[index];
+                }
+                if (!indentation_.empty())
+                    *sout_ << " ";
+                *sout_ << "]";
+            }
+        }
+    }
+
+    bool BuiltStyledStreamWriter::isMultilineArray(Value const &value) {
+        ArrayIndex const size = value.size();
+        bool isMultiLine = size * 3 >= rightMargin_;
+        childValues_.clear();
+        for (ArrayIndex index = 0; index < size && !isMultiLine; ++index) {
+            Value const &childValue = value[index];
+            isMultiLine = ((childValue.isArray() || childValue.isObject()) &&
+                           !childValue.empty());
+        }
+        if (!isMultiLine) // check if line length > max line length
+        {
+            childValues_.reserve(size);
+            addChildValues_ = true;
+            ArrayIndex lineLength = 4 + (size - 1) * 2; // '[ ' + ', '*n + ' ]'
+            for (ArrayIndex index = 0; index < size; ++index) {
+                if (hasCommentForValue(value[index])) {
+                    isMultiLine = true;
+                }
+                writeValue(value[index]);
+                lineLength += static_cast<ArrayIndex>(childValues_[index].length());
+            }
+            addChildValues_ = false;
+            isMultiLine = isMultiLine || lineLength >= rightMargin_;
+        }
+        return isMultiLine;
+    }
+
+    void BuiltStyledStreamWriter::pushValue(String const &value) {
+        if (addChildValues_)
+            childValues_.push_back(value);
+        else
+            *sout_ << value;
+    }
+
+    void BuiltStyledStreamWriter::writeIndent() {
+        // blep intended this to look at the so-far-written string
+        // to determine whether we are already indented, but
+        // with a stream we cannot do that. So we rely on some saved state.
+        // The caller checks indented_.
+
+        if (!indentation_.empty()) {
+            // In this case, drop newlines too.
+            *sout_ << '\n' << indentString_;
+        }
+    }
+
+    void BuiltStyledStreamWriter::writeWithIndent(String const &value) {
+        if (!indented_)
+            writeIndent();
+        *sout_ << value;
+        indented_ = false;
+    }
+
+    void BuiltStyledStreamWriter::indent() { indentString_ += indentation_; }
+
+    void BuiltStyledStreamWriter::unindent() {
+        assert(indentString_.size() >= indentation_.size());
+        indentString_.resize(indentString_.size() - indentation_.size());
+    }
+
+    void BuiltStyledStreamWriter::writeCommentBeforeValue(Value const &root) {
+        if (cs_ == CommentStyle::None)
+            return;
+        if (!root.hasComment(commentBefore))
+            return;
+
+        if (!indented_)
+            writeIndent();
+        const String &comment = root.getComment(commentBefore);
+        String::const_iterator iter = comment.begin();
+        while (iter != comment.end()) {
+            *sout_ << *iter;
+            if (*iter == '\n' && ((iter + 1) != comment.end() && *(iter + 1) == '/'))
+                // writeIndent();  // would write extra newline
+                *sout_ << indentString_;
+            ++iter;
+        }
+        indented_ = false;
+    }
+
+    void BuiltStyledStreamWriter::writeCommentAfterValueOnSameLine(
+            Value const &root) {
+        if (cs_ == CommentStyle::None)
+            return;
+        if (root.hasComment(commentAfterOnSameLine))
+            *sout_ << " " + root.getComment(commentAfterOnSameLine);
+
+        if (root.hasComment(commentAfter)) {
+            writeIndent();
+            *sout_ << root.getComment(commentAfter);
+        }
+    }
+
+// static
+    bool BuiltStyledStreamWriter::hasCommentForValue(const Value &value) {
+        return value.hasComment(commentBefore) ||
+               value.hasComment(commentAfterOnSameLine) ||
+               value.hasComment(commentAfter);
+    }
+
+///////////////
+// StreamWriter
+
+    StreamWriter::StreamWriter() : sout_(nullptr) {}
+
+    StreamWriter::~StreamWriter() = default;
+
+    StreamWriter::Factory::~Factory() = default;
+
+    StreamWriterBuilder::StreamWriterBuilder() { setDefaults(&settings_); }
+
+    StreamWriterBuilder::~StreamWriterBuilder() = default;
+
+    StreamWriter *StreamWriterBuilder::newStreamWriter() const {
+        const String indentation = settings_["indentation"].asString();
+        const String cs_str = settings_["commentStyle"].asString();
+        const String pt_str = settings_["precisionType"].asString();
+        const bool eyc = settings_["enableYAMLCompatibility"].asBool();
+        const bool dnp = settings_["dropNullPlaceholders"].asBool();
+        const bool usf = settings_["useSpecialFloats"].asBool();
+        const bool emitUTF8 = settings_["emitUTF8"].asBool();
+        unsigned int pre = settings_["precision"].asUInt();
+        CommentStyle::Enum cs = CommentStyle::All;
+        if (cs_str == "All") {
+            cs = CommentStyle::All;
+        } else if (cs_str == "None") {
+            cs = CommentStyle::None;
+        } else {
+            throwRuntimeError("commentStyle must be 'All' or 'None'");
+        }
+        PrecisionType precisionType(significantDigits);
+        if (pt_str == "significant") {
+            precisionType = PrecisionType::significantDigits;
+        } else if (pt_str == "decimal") {
+            precisionType = PrecisionType::decimalPlaces;
+        } else {
+            throwRuntimeError("precisionType must be 'significant' or 'decimal'");
+        }
+        String colonSymbol = " : ";
+        if (eyc) {
+            colonSymbol = ": ";
+        } else if (indentation.empty()) {
+            colonSymbol = ":";
+        }
+        String nullSymbol = "null";
+        if (dnp) {
+            nullSymbol.clear();
+        }
+        if (pre > 17)
+            pre = 17;
+        String endingLineFeedSymbol;
+        return new BuiltStyledStreamWriter(indentation, cs, colonSymbol, nullSymbol,
+                                           endingLineFeedSymbol, usf, emitUTF8, pre,
+                                           precisionType);
+    }
+
+    bool StreamWriterBuilder::validate(Json::Value *invalid) const {
+        static const auto &valid_keys = *new std::set<String>{
+                "indentation",
+                "commentStyle",
+                "enableYAMLCompatibility",
+                "dropNullPlaceholders",
+                "useSpecialFloats",
+                "emitUTF8",
+                "precision",
+                "precisionType",
+        };
+        for (auto si = settings_.begin(); si != settings_.end(); ++si) {
+            auto key = si.name();
+            if (valid_keys.count(key))
+                continue;
+            if (invalid)
+                (*invalid)[std::move(key)] = *si;
+            else
+                return false;
+        }
+        return invalid ? invalid->empty() : true;
+    }
+
+    Value &StreamWriterBuilder::operator[](const String &key) {
+        return settings_[key];
+    }
+
+// static
+    void StreamWriterBuilder::setDefaults(Json::Value *settings) {
+        //! [StreamWriterBuilderDefaults]
+        (*settings)["commentStyle"] = "All";
+        (*settings)["indentation"] = "\t";
+        (*settings)["enableYAMLCompatibility"] = false;
+        (*settings)["dropNullPlaceholders"] = false;
+        (*settings)["useSpecialFloats"] = false;
+        (*settings)["emitUTF8"] = false;
+        (*settings)["precision"] = 17;
+        (*settings)["precisionType"] = "significant";
+        //! [StreamWriterBuilderDefaults]
+    }
+
+    String writeString(StreamWriter::Factory const &factory, Value const &root) {
+        OStringStream sout;
+        StreamWriterPtr const writer(factory.newStreamWriter());
+        writer->write(root, &sout);
+        return sout.str();
+    }
+
+    OStream &operator<<(OStream &sout, Value const &root) {
+        StreamWriterBuilder builder;
+        StreamWriterPtr const writer(builder.newStreamWriter());
+        writer->write(root, &sout);
+        return sout;
+    }
+
+} // namespace Json
+
+// //////////////////////////////////////////////////////////////////////
+// End of content of file: src/lib_json/json_writer.cpp
+// //////////////////////////////////////////////////////////////////////
+
+
+
+
+

+ 37 - 0
library_core/src/main/cpp/yyxxgame.cpp

@@ -0,0 +1,37 @@
+//
+// Created by KaiWeiCai on 2022/1/14.
+//
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+void InitSdkDrive(JNIEnv *env, jobject clz, jobject context) {
+//    CommMap::GetInstance()->Init(env, context);
+}
+
+static JNINativeMethod gMethods[] = {
+        {"initDrive",  "(Landroid/content/Context;)V",            (void *) InitSdkDrive},
+//        {"setParam",   "(Ljava/lang/String;Ljava/lang/String;)V", (void *) SetParam},
+//        {"getParam",   "(Ljava/lang/String;)Ljava/lang/String;",  (void *) GetParam},
+//        {"invokeJob",  "(Ljava/lang/String;)Ljava/lang/String;",  (void *) InvokeJob},
+//        {"parseJob",   "(Ljava/lang/String;)Ljava/lang/String;",  (void *) ParseJob},
+//        {"RsaEncrypt", "(Ljava/lang/String;)Ljava/lang/String;",  (void *) RsaEncrypt}
+};
+
+JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {
+    JNIEnv *env = nullptr;
+    if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {
+        return JNI_ERR;
+    }
+    jclass clz = env->FindClass("cn/yyxx/commsdk/core/NTest");
+    if (env->RegisterNatives(clz, gMethods, sizeof(gMethods) / sizeof(gMethods[0])) < 0) {
+        return JNI_ERR;
+    }
+    return JNI_VERSION_1_6;
+}
+
+#ifdef __cplusplus
+}
+#endif

+ 2 - 0
library_core/src/main/java/cn/yyxx/commsdk/core/NTest.java

@@ -0,0 +1,2 @@
+package cn.yyxx.commsdk.core;public class NTest {
+}

+ 2 - 0
library_core/src/main/java/cn/yyxx/commsdk/core/network/Hosts.java

@@ -0,0 +1,2 @@
+package cn.yyxx.commsdk.core.network;public class Hosts {
+}

+ 2 - 0
library_core/src/main/java/cn/yyxx/commsdk/core/network/SdkRequest.java

@@ -0,0 +1,2 @@
+package cn.yyxx.commsdk.core.network;public class SdkRequest {
+}

+ 165 - 0
library_core/src/main/java/cn/yyxx/commsdk/core/network/VolleyRequest.java

@@ -0,0 +1,165 @@
+package cn.qingshi.gamesdk.core.network;
+
+import android.content.Context;
+import android.text.TextUtils;
+
+import org.json.JSONException;
+import org.json.JSONObject;
+
+import java.io.IOException;
+import java.util.HashMap;
+import java.util.Map;
+
+import cn.qingshi.gamesdk.base.utils.Logger;
+import cn.qingshi.gamesdk.core.entity.ResultInfo;
+import cn.qingshi.gamesdk.core.internal.IFileRequestCallback;
+import cn.qingshi.gamesdk.core.internal.IRequestCallback;
+import cn.yyxx.support.FileUtils;
+import cn.yyxx.support.JsonUtils;
+import cn.yyxx.support.volley.VolleySingleton;
+import cn.yyxx.support.volley.source.DefaultRetryPolicy;
+import cn.yyxx.support.volley.source.NetworkResponse;
+import cn.yyxx.support.volley.source.ParseError;
+import cn.yyxx.support.volley.source.Request;
+import cn.yyxx.support.volley.source.Response;
+import cn.yyxx.support.volley.source.VolleyError;
+import cn.yyxx.support.volley.source.toolbox.HttpHeaderParser;
+import cn.yyxx.support.volley.source.toolbox.JsonObjectRequest;
+import cn.yyxx.support.volley.source.toolbox.JsonRequest;
+
+/**
+ * @author #Suyghur.
+ * Created on 2021/04/28
+ */
+public class VolleyRequest {
+
+    /**
+     * 超时时间
+     */
+    private static final int MAX_TIMEOUT = 10 * 1000;
+
+
+    public static void post(Context context, String url, JSONObject jsonObject, final IRequestCallback callback) {
+        try {
+//            StringBuilder raw = new StringBuilder();
+//            raw.append("access_token=").append(jsonObject.getString("access_token"))
+//                    .append("&time=").append(jsonObject.getString("time"))
+//                    .append("&context=").append(jsonObject.getString("context"))
+//                    .append("&appkey=").append(ParamsUtils.getQSAppKey(context));
+//
+//            Logger.d("raw : " + raw);
+//            String sign = Md5Utils.encodeByMD5(raw.toString().toLowerCase());
+//            jsonObject.put("sign", sign);
+            Logger.logHandler("请求地址 : " + url + "\n");
+            Logger.logHandler("请求参数 : " + jsonObject.toString() + "\n");
+            Logger.d("请求地址 : " + url);
+            Logger.d("请求参数 : " + jsonObject.toString());
+
+            JsonRequest<JSONObject> request = new JsonObjectRequest(Request.Method.POST, url, jsonObject, new Response.Listener<JSONObject>() {
+                @Override
+                public void onResponse(JSONObject response) {
+                    ResultInfo resultInfo = new ResultInfo();
+                    if (response != null) {
+                        try {
+                            resultInfo.code = response.getInt("code");
+                            resultInfo.msg = response.getString("message");
+                            if (JsonUtils.hasJsonKey(response, "data")) {
+                                resultInfo.data = response.getString("data");
+                            } else {
+                                resultInfo.data = "";
+                            }
+                        } catch (JSONException e) {
+                            e.printStackTrace();
+                            resultInfo.code = -1;
+                            resultInfo.msg = "数据解析异常";
+                            resultInfo.data = "";
+                        }
+                    } else {
+                        resultInfo.code = -1;
+                        resultInfo.msg = "接口请求异常";
+                        resultInfo.data = "";
+                    }
+                    Logger.logHandler("返回内容 : " + resultInfo.toString());
+                    Logger.d("onResponse : " + resultInfo.toString());
+                    if (callback != null) {
+                        callback.onResponse(resultInfo);
+                    }
+                }
+            }, new Response.ErrorListener() {
+                @Override
+                public void onErrorResponse(VolleyError volleyError) {
+                    Logger.e("onErrorResponse : " + volleyError.toString());
+                    if (callback != null) {
+                        callback.onResponse(getErrorResultInfo(volleyError));
+                    }
+                }
+            }) {
+                @Override
+                public Map<String, String> getHeaders() {
+                    HashMap<String, String> headers = new HashMap<>();
+                    headers.put("Accept", "application/json");
+                    headers.put("Content-Type", "application/json;charset=UTF-8");
+                    return headers;
+                }
+            };
+            request.setRetryPolicy(new DefaultRetryPolicy(MAX_TIMEOUT, 1, 1.0f));
+            VolleySingleton.getInstance(context.getApplicationContext()).addToRequestQueue(context.getApplicationContext(), request);
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+    }
+
+    public static void downloadImageFile(Context context, String url, final String filePath, final IFileRequestCallback callback) {
+        if (TextUtils.isEmpty(filePath)) {
+            callback.onResult(-1, "file path is empty !!!");
+            return;
+        }
+        Request<byte[]> request = new Request<byte[]>(Request.Method.GET, url, new Response.ErrorListener() {
+            @Override
+            public void onErrorResponse(VolleyError volleyError) {
+                callback.onResult(-1, volleyError.getLocalizedMessage());
+            }
+        }) {
+
+            @Override
+            protected Response<byte[]> parseNetworkResponse(NetworkResponse networkResponse) {
+                try {
+                    if (networkResponse.data == null) {
+                        return Response.error(new ParseError(networkResponse));
+                    } else {
+                        return Response.success(networkResponse.data, HttpHeaderParser.parseCacheHeaders(networkResponse));
+                    }
+                } catch (OutOfMemoryError e) {
+                    e.printStackTrace();
+                    return Response.error(new ParseError(e));
+                }
+            }
+
+            @Override
+            protected void deliverResponse(byte[] bytes) {
+                Logger.d("volley download image file success, start to save file ...");
+                try {
+                    FileUtils.saveFile(filePath, bytes);
+                    callback.onResult(0, "缓存文件成功");
+                } catch (IOException e) {
+                    e.printStackTrace();
+                    callback.onResult(0, "缓存文件失败");
+                }
+            }
+        };
+        VolleySingleton.getInstance(context.getApplicationContext()).addToRequestQueue(context.getApplicationContext(), request);
+    }
+
+    private static ResultInfo getErrorResultInfo(VolleyError volleyError) {
+        ResultInfo resultInfo = new ResultInfo();
+        if (volleyError.networkResponse != null) {
+            resultInfo.code = volleyError.networkResponse.statusCode;
+            resultInfo.msg = volleyError.getMessage();
+        } else {
+            resultInfo.code = 400;
+            resultInfo.msg = "网络异常";
+        }
+        resultInfo.data = "";
+        return resultInfo;
+    }
+}